STL之迭代器失效问题

原创 2015年07月06日 16:44:28

1 常见的STL容器

标准STL序列容器vectorstringdequelist

标准STL关联容器setmultisetmapmultimap

非标准序列容器slistslist是一个单向链表

非标准关联容器hash_sethash_multisethash_maphash_multimap

几种标准非STL容器stackqueuepriority_queue


从迭代器的性质出发,上述容器可分为三类:

1) vector, string类

2) deque类

3) list类

slist, setmultiset, mapmultimap基于rb_tree实现,迭代器性质也类是list.

同理, hash_sethash_multisethash_maphash_multimap也是如此.因为它们的底层是hashtable,而它采用的是开链法.

3) 适配器类

stack, queue,priority_queue,这类有特殊的存取规则,不提供的迭代器.

正因如此, 所以C++ primer一书只说了vector, string, list(forward_list)和deque的迭代器失效问题.


2 添加元素队迭代器的影响

2.1 vector, string类

1)存储空间重新分配

指向容器的迭代器,指针,引用都将失效.(所有空间重新分配, 之前的空间被释放)

2)存储空间未重新分配

插入点之前的迭代器,指针,引用有效,但插入点之后的都将无效.(发生了插入点后元素位置的移动)


2.2 deque类

1) 首尾位置之外插入

指向容器的迭代器,指针,引用都将失效.这与vector不同(保证插入点之前有效).这与deque的实现机制有关.

deque的内存布局


deque插入源码



因为, deque的插入为根据插入前前后元素的数量,选择一个较小的进行移动,所以具体在一次插入操作中移动的是前面还是后面部分不能确定.

所以并没有保持像list的性质.

2) 首尾位置插入

迭代器会失效, 但指向存在元素的引用和指针不会失效.

这是C++ primer的原话.但直至没有想清楚为什么?刚开始我的理解是,首(尾)插入只会影响插入前的首(尾)位置的迭代器,其他位置的迭代器应该仍有效.

即使插入操作造成map的重新分配和拷贝或缓存节点的增加.但原deque除首(尾)位置,并没有改变.

后来发现还是我错啦.药弄明白还得从deque的迭代器的实现原理入手.以下是个人的理解.不对的地方还请读者指出.


首尾插入引起的deque迭代器失效,是由map重新分配,引起原来所有迭代器的node指针失效导致的,

可以将这种失效看作是一种广义的失效.因为这种迭代器负责的是一个缓存节点.即使迭代器失效,

但在迭代器内部仍然有效, 即局部有效.

如果map没有重新分配,除原首尾位置,其他迭代器应该仍是有效的.但是对于使用者而言,你并不知道何时没有发生map重新分配.

因此也作失效处理.


2.3 list类

指向容器的迭代器(包括首尾迭代器), 指针, 引用仍有效.


3 删除元素队迭代的影响

删除元素,一定会使指向删除节点的迭代器失效.

3.1 vector, string类

删除点之前的迭代器, 指针, 引用仍有效.但当我们删除元素时,尾后迭代器总是失效.

3.2 deque类

1)首尾之外节点删除, 所有原迭代器都失效.(原因和插入相同)

2)首尾节点删除, 其他迭代器,指针和引用仍会有效. 

3.3 list类

指向容器的其他迭代器,指针和引用仍会有效.





版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

STL中迭代器失效问题

1)vector erase造成迭代器失效  erase成员函数,它删除了迭代器指向的元素,并且返回要被删除的元素之后的迭代器。 #include #include using...
  • oniy_
  • oniy_
  • 2015-05-27 15:54
  • 528

STL容器迭代器失效问题

众所周知当使用一个容器的insert或者erase函数通过迭代器插入或删除元素"可能"会导致迭代器失效,因此很多建议都是让我们获取insert或者erase返回的迭代器,以便用重新获取新的有效的迭代器...

STL迭代器失效问题

最近在项目开发中,遇到一个异常,经过测试,发现是迭代器失效问题,于是稍微总结了一下。 vector迭代器失效测试: 测试程序: void vectorTest() {     ...

STL容器的遍历插入或删除(迭代器失效问题的统一解决)

STL容器根据迭代器的失效问题,其实可以分为两类容器: (1)数组型容器的插入删除操作:vector、string、deque(均为顺序存储)         由于这类容器的插入或删除都会使所有迭...

STL迭代器失效问题

vector迭代器失效测试: 测试程序: void vectorTest() { vector container; for (int i = 0; i < 10; i++) ...

c++迭代器失效问题

  • 2015-10-13 17:14
  • 165KB
  • 下载

STL迭代器及迭代器失效问题

迭代器失效: 典型的迭代器失效. 首先对于vector而言,添加和删除操作可能使容器的部分或者全部迭代器失效。那为什么迭代器会失效呢?vector元素在内存中是顺序存储,试想:如果当前容器中已...

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

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

STL容器迭代器失效浅析

迭代器失效一般发生在对容器进行插入及删除操作时,插入/删除操作可能导致空间的重配置以及所指对象的位移而带来迭代器失效问题,我们可以归纳为以下两点: 由于容器元素整体“迁移”导致存放原容器元素的空间不再...

STL容器特征总结和迭代器失效

Vector 内部数据结构:连续存储,例如数组。随机访问每个元素,所需要的时间为常量。在末尾增加或删除元素所需时间与元素数目无关,在中间或开头增加或删除元素所需时间随元素数目呈线性变化。可动态增加或...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)