STL map 内存改变,迭代器失效,_Isnil(_Ptr)和红黑树

最近在做项目时发现一个crash的问题,当时得到的dmp文件显示crash在一个以map为循环变量的循环中,crash位置在如下的代码中标出。

void _Inc()
            {    // move to node with next larger value

#if _HAS_ITERATOR_DEBUGGING
            if (this->_Mycont == 0
                || _Ptr == 0
                || _Isnil(_Ptr))
                {
                _DEBUG_ERROR("map/set iterator not incrementable");
                _SCL_SECURE_OUT_OF_RANGE;
                }
 #else
            _SCL_SECURE_VALIDATE(this->_Has_container());
            if (_Isnil(_Ptr))---------------------------------------->Why crash here?
                {
                _SCL_SECURE_OUT_OF_RANGE;
                // end() shouldn't be incremented, don't move if _SCL_SECURE is not turned on
                }
 #endif /* _HAS_ITERATOR_DEBUGGING */

else if (!_Isnil(_Right(_Ptr)))
                _Ptr = _Min(_Right(_Ptr));    // ==> smallest of right subtree
            else
                {    // climb looking for right subtree
                _Nodeptr _Pnode;
                while (!_Isnil(_Pnode = _Parent(_Ptr))
                    && _Ptr == _Right(_Pnode))
                    _Ptr = _Pnode;    // ==> parent while right subtree
                _Ptr = _Pnode;    // ==> parent (head if end())
                }
            }

这是C++ 中红黑树迭代器的标准实现,那从这个栈帧能说明我们的代码哪出问题了么?在阅读红黑树的实现代码中有一条语句困扰了我大约半个小时的时间,这条语句就是:

_Isnil(_Ptr)

标准实现中到处都是这条语句,2年前算法导论系统的学习过一遍,但是由于长时间没有相关的功能需要用到这么深入的知识,有些具体的问题已经记得不是很清楚,于是为了弄对这条语句以及对红黑树有透测的理解,再一次对红黑树知识进行了系统的学习,并翻阅了一些资料,为了使本文自成体系,下面对基础知识进行一些说明。

红黑树定义

红黑树是一种自平衡二叉查找树,是在计算机科学中用到的一种数据结构,典型的用途是实现关联数组,如STL map 和set,和普通二叉树相比它的实现上稍微有些复杂,但它的操作有着良好的最坏情况运行时间,并且在实践中是高效的: 它可以在O(log n)时间内做查找,插入和删除,这里的n是树中元素的数目。

红黑树性质

红黑树是每个节点都带有颜色属性的二叉查找树,颜色为红色或黑色。在二叉查找树强制一般要求以外,对于任何有效的红黑树我们增加了如下的额外要求:

性质1. 节点是红色或黑色。

性质2. 根是黑色。

性质3. 所有叶子都是黑色(叶子是NIL节点)。

性质4. 每个红色节点的两个子节点都是黑色。(从每个叶子到根的所有路径上不能有两个连续的红色节点)

性质5. 从任一节点到其每个叶子的所有简单路径都包含相同数目的黑色节点。

算法导论原版定义

A red-black tree is a binary search tree with one extra bit of storage per node: its color, which can be either RED or BLACK. By constraining the node colors on any simple path from the root to a leaf, red-black trees ensure that no such path is more than twice as long as any other, so that the tree is approximately balanced. Each node of the tree now contains the attributes color, key, left, right, and p. If a child or the parent of a node does not exist, the corresponding pointer attribute of the node contains the value NIL. We shall regard these NILs as being pointers to leaves (external nodes) of the binary search tree and the normal, key-bearing nodes as being internal nodes of the tree. A red-black tree is a binary tree that satisfies the following red-black properties:

1. Every node is either red or black.

2. The root is black.

3. Every leaf (NIL) is black.

4. If a node is red, then both its children are black.

5. For each node, all simple paths from the node to descendant leaves contain the same number of black nodes.

典型的红黑树如下图所示

红黑树的典型操作

插入/删除/左旋/右旋/查找/遍历再次不做赘述,有兴趣的读者可以参考STL源码实现。

总结

再来看看我们的问题,请看如下的迭代器实现,迭代器++操作符会调用_Inc(),由于多线程,map被破坏,迭代器失效导致循环无法结束并crash。那么为什么会crash呢?原来红黑树的标准实现中真正的叶子节点都是NIL(哨兵)节点,并且规定叶子节点已经到达了红黑树的边界,所以不能在++,如果在++系统就会 crash 。

const_iterator& operator++()

{      // preincrement

_Inc();

return (*this);

}

那么这条语句:_Isnil(_Ptr) 到底是什么意思呢?

答案:就是说当前操作的迭代器是否已经指向了黑色的叶子节点,这个节点其实是哨兵(NIL)节点,指向了这个哨兵节点以后的迭代器是不能++的。

转载于:https://www.cnblogs.com/pugang/archive/2012/06/22/2558542.html

STL map 内存改变,迭代器失效,_Isnil(_Ptr)和红黑树相关推荐

  1. STL源码剖析---迭代器失效小结

    迭代器(iterator)是一个可以对其执行类似指针的操作(如:解除引用(operator*())和递增(operator++()))的对象,我们可以将它理解成为一个指针.但它又不是我们所谓普通的指针 ...

  2. STL的erase()陷阱-迭代器失效总结

    下面材料整理自Internet&著作. STL中的容器按存储方式分为两类,一类是按以数组形式存储的容器(如:vector .deque):另一类是以不连续的节点形式存储的容器(如:list.s ...

  3. STL中容器vector迭代器失效的相关问题

    迭代器失效,有两个层面的意思: 无法通过迭代器++,–操作遍历整个stl容器.记作: 第一层失效. 无法通过迭代器存取迭代器所指向的内存. 记作: 第二层失效. vector是个连续内存存储的容器,如 ...

  4. map怎么转化dto_阿里面试题:为什么Map桶中个数超过8才转为红黑树

    点击上方"linkoffer", 选择关注公众号高薪职位第一时间送达 这是笔者一个好友面试阿里时,被问及的一个问题,应该不少人看到这个问题都会一面懵逼.因为,大部分的文章都是分析链 ...

  5. Java多线程学习二十二:为什么 Map 桶中超过 8 个才转为红黑树

    为什么 Map 的桶中超过 8 个才转为红黑树? JDK 1.8 的 HashMap 和 ConcurrentHashMap 都有这样一个特点:最开始的 Map 是空的,因为里面没有任何元素,往里放元 ...

  6. map怎么转化dto_阿里面试:为什么Map桶中个数超过8才转为红黑树

    这是一个好友面试阿里时,被问到的一个问题,应该不少人看到这个问题都会一面懵逼.因为,大部分的文章都是分析链表是怎么转换成红黑树的,但是并没有说明为什么当链表长度为8的时候才做转换动作.第一反应也是一样 ...

  7. 精选(1)为什么Map桶中个数超过8才转为红黑树

    这是一个好友面试阿里时,被问到的一个问题,应该不少人看到这个问题都会一面懵逼.因为,大部分的文章都是分析链表是怎么转换成红黑树的,但是并没有说明为什么当链表长度为8的时候才做转换动作.第一反应也是一样 ...

  8. 为什么Map桶中个数超过8才转为红黑树

    直白一点:就是trade-off,空间和时间上的权衡! 源码中有如下内容: * Because TreeNodes are about twice the size of regular nodes, ...

  9. 红黑树实现——STL中的map

    From: http://blog.csdn.net/zhongjiekangping/article/details/6934571 红黑树实现--STL中的map [ 2009-07-24 13: ...

最新文章

  1. 0UL能做C语言常量吗,c语言中的0UL或1UL是什么意思
  2. Mac下批量删除.svn文件
  3. python字符串三种常用的方法或函数_python中字符串常用的函数
  4. 反编译后怎么修改服务器地址,反编译后怎么修改服务器地址
  5. ArcGIS Server 10.1发布结果地图服务——与10.0的区别及过程
  6. html5d调用百度语音,易语言调用百度语音平台实现文字转换语音功能的代码
  7. 在SQLite数据库中获取新插入数据自增长的ID值
  8. linux下使用source执行文件的含义
  9. QtWebApp的使用【在Qt中搭建HTTP服务器】(一)
  10. 随笔:《像火箭科学家一样思考:将不可能变成可能》观书有感
  11. 木吉他学习的高频问答(持续更新)
  12. 关于雨林木风版的linux操作系统ymlf_os_3.0
  13. Nginx下的反向代理 双层代理 负载均衡
  14. Windows 2003环境下IIS下ASP+ACCESS的配置方法
  15. 答案揭晓 | 关于云基础架构,你了解多少?
  16. 关于ntko从后台传输文档时发生文件存取错误,暨关于response使用的注意点
  17. 端午安康,用python给你画盘粽子~啾啾
  18. opencv级联分类器(linux系统)
  19. from origin ‘null‘ has been blocked by CORS policy: Cross origin requests are only supported for ...
  20. sw标准件不能配置_sw标准件库调不出来(在使用solidworks2009标准件库 调不出零件)...

热门文章

  1. StringBuffer类的功能
  2. 使用 putty 免密码 ssh 登录 mac os
  3. java----单例模式
  4. 基于用户的协同过滤算法
  5. Oracle分页小谈
  6. 用oracle的java存储过程实现BLOB字段的字符串读取
  7. AUCAD自定义[2006.9.22]
  8. 软件测试 学习之路 html基础
  9. 狱警讲述死刑_BDFL是死刑吗?
  10. 记录是一段旅程:记录Scribus可获得3课