c++迭代器失效的情况汇总

一、序列式容器(数组式容器)

对于序列式容器(如vector,deque),序列式容器就是数组式容器,删除当前的iterator会使后面所有元素的iterator都失效。这是因为vetor,deque使用了连续分配的内存,删除一个元素导致后面所有的元素会向前移动一个位置。所以不能使用erase(iter++)的方式,还好erase方法可以返回下一个有效的iterator。

for (iter = cont.begin(); iter != cont.end();)
{
  (*it)->doSomething();
  if (shouldDelete(*iter))
   iter = cont.erase(iter); //erase删除元素,返回下一个迭代器
  else
   ++iter;
}

迭代器失效:

void vectorTest()
{
  vector<int> container;
  for (int i = 0; i < 10; i++)
  {
    container.push_back(i);
  }

  vector<int>::iterator iter;
   for (iter = container.begin(); iter != container.end(); iter++)
  {
      if (*iter > 3)
       container.erase(iter);
  }

   for (iter = container.begin(); iter != container.end(); iter++)
  {
      cout<<*iter<<endl;
  }
}

报错是:vectoriterator not incrementable.

迭代器在执行++操作时报错!已经失效的迭代器不能再进行自增运算了。++代码大致实现如下:

_Myiter operator++(int)
{
  _Myiter _Tmp=*this;
  ++*this;
  return (_Tmp);
}

对于序列式容器,比如vector,删除当前的iterator会使后面所有元素的iterator都失效。这是因为顺序容器内存是连续分配(分配一个数组作为内存),删除一个元素导致后面所有的元素会向前移动一个位置。(删除了一个元素,该元素后面的所有元素都要挪位置,所以,iter++,已经指向的是未知内存)。

但是erase方法可以返回下一个有效的iterator。所以代码做如下修改,就OK了。

void vectorTest()
{
  vector<int> container;
  for (int i = 0; i < 10; i++)
  {
    container.push_back(i);
  }

  vector<int>::iterator iter;
  for (iter = container.begin(); iter != container.end();)
  {
      if (*iter > 3) {
        iter = container.erase(iter);
      }
      else {
        iter ++;
      }

  }

  for (iter = container.begin(); iter != container.end(); iter++)
  {
      cout<<*iter<<endl;
  }
}

总结:vector是一个顺序容器,在内存中是一块连续的内存,当删除一个元素后,内存中的数据会发生移动,以保证数据的紧凑。所以删除一个数据后,其他数据的地址发生了变化,之前获取的迭代器根据原有的信息就访问不到正确的数据。

所以为了防止vector迭代器失效,常用如下方法:

for (iter = container.begin(); iter != container.end(); )
{
      if (*iter > 3)
       iter = container.erase(iter);  //erase的返回值是删除元素下一个元素的迭代器
      else{
        iter++;
      }
}

这样删除后iter指向的元素后,返回的是下一个元素的迭代器,这个迭代器是vector内存调整过后新的有效的迭代器。

二、关联式容器

对于关联容器(如map, set,multimap,multiset),删除当前的iterator,仅仅会使当前的iterator失效,只要在erase时,递增当前iterator即可。这是因为map之类的容器,使用了红黑树来实现,插入、删除一个结点不会对其他结点造成影响。erase迭代器只是被删元素的迭代器失效,但是返回值为void,所以要采用erase(iter++)的方式删除迭代器。

for (iter = cont.begin(); it != cont.end();)
{
  (*iter)->doSomething();
  if (shouldDelete(*iter))
   cont.erase(iter++);
  else
   ++iter;
}

//测试错误的Map删除元素
void mapTest()
{
  map<int, string> dataMap;

  for (int i = 0; i < 100; i++)
  {
      string strValue = "Hello, World";

      stringstream ss;
      ss<<i;
      string tmpStrCount;
      ss>>tmpStrCount;
      strValue += tmpStrCount;
      dataMap.insert(make_pair(i, strValue));
  }

  cout<<"MAP元素内容为:"<<endl;
   map<int, string>::iterator iter;
  for (iter = dataMap.begin(); iter != dataMap.end(); iter++)
  {
      int nKey = iter->first;
      string strValue = iter->second;
      cout<<strValue<<endl;
  }

  cout<<"内容开始删除:"<<endl;
  //删除操作引发迭代器失效
  for (iter = dataMap.begin(); iter != dataMap.end();iter++)
  {
      int nKey = iter->first;
      string strValue = iter->second;

      if (nKey % 2 == 0)
      {
        dataMap.erase(iter);  //错误

      }
      /* cout<<iter->second<<endl;*/
  }
}

出错:

解析:dataMap.erase(iter)之后,iter就已经失效了,所以iter无法自增,即iter++就会出bug.解决方案,就是在iter失效之前,先自增。

void mapTest()
{
  map<int, string> dataMap;

  for (int i = 0; i < 100; i++)
  {
      string strValue = "Hello, World";

      stringstream ss;
      ss<<i;
      string tmpStrCount;
      ss>>tmpStrCount;
      strValue += tmpStrCount;
      dataMap.insert(make_pair(i, strValue));
  }

  cout<<"MAP元素内容为:"<<endl;
  map<int, string>::iterator iter;
  for (iter = dataMap.begin(); iter != dataMap.end(); iter++)
  {
      int nKey = iter->first;
      string strValue = iter->second;
      cout<<strValue<<endl;
  }

  cout<<"内容开始删除:"<<endl;
  for (iter = dataMap.begin(); iter != dataMap.end();)
  {
      int nKey = iter->first;
      string strValue = iter->second;

      if (nKey % 2 == 0)
      {
        dataMap.erase(iter++);
        auto a = iter;

      }
      else {
        iter ++;
      }
  }
}

解析:dataMap.erase(iter++);这句话分三步走,先把iter传值到erase里面,然后iter自增,然后执行erase,所以iter在失效前已经自增了。

map是关联容器,以红黑树或者平衡二叉树组织数据,虽然删除了一个元素,整棵树也会调整,以符合红黑树或者二叉树的规范,但是单个节点在内存中的地址没有变化,变化的是各节点之间的指向关系。

所以在map中为了防止迭代器失效,在有删除操作时,常用如下方法:

for (iter = dataMap.begin(); iter != dataMap.end(); )
{
     int nKey = iter->first;
     string strValue = iter->second;

     if (nKey % 2 == 0)
     {
        map<int, string>::iterator tmpIter = iter;
      iter++;
        dataMap.erase(tmpIter);
        //dataMap.erase(iter++) 这样也行

     }else
   {
     iter++;
     }
}

三、链表式容器

对于链表式容器(如list),删除当前的iterator,仅仅会使当前的iterator失效,这是因为list之类的容器,使用了链表来实现,插入、删除一个结点不会对其他结点造成影响。只要在erase时,递增当前iterator即可,并且erase方法可以返回下一个有效的iterator。

方式一:递增当前iterator

for (iter = cont.begin(); it != cont.end();)
{
  (*iter)->doSomething();
  if (shouldDelete(*iter))
   cont.erase(iter++);
  else
   ++iter;
}

方式二:通过erase获得下一个有效的iterator

for (iter = cont.begin(); iter != cont.end();)
{
  (*it)->doSomething();
  if (shouldDelete(*iter))
   iter = cont.erase(iter); //erase删除元素,返回下一个迭代器
  else
   ++iter;
}

四、总结

迭代器失效分三种情况考虑,也是分三种数据结构考虑,分别为数组型,链表型,树型数据结构。

数组型数据结构:该数据结构的元素是分配在连续的内存中,insert和erase操作,都会使得删除点和插入点之后的元素挪位置,所以,插入点和删除掉之后的迭代器全部失效,也就是说insert(*iter)(或erase(*iter)),然后在iter++,是没有意义的。解决方法:erase(*iter)的返回值是下一个有效迭代器的值。 iter =cont.erase(iter);

链表型数据结构:对于list型的数据结构,使用了不连续分配的内存,删除运算使指向删除位置的迭代器失效,但是不会失效其他迭代器.解决办法两种,erase(*iter)会返回下一个有效迭代器的值,或者erase(iter++).

树形数据结构: 使用红黑树来存储数据,插入不会使得任何迭代器失效;删除运算使指向删除位置的迭代器失效,但是不会失效其他迭代器.erase迭代器只是被删元素的迭代器失效,但是返回值为void,所以要采用erase(iter++)的方式删除迭代器。

注意:经过erase(iter)之后的迭代器完全失效,该迭代器iter不能参与任何运算,包括iter++,*ite

以上就是c++迭代器失效的情况汇总的详细内容,更多关于c++迭代器失效的资料请关注我们其它相关文章!

(0)

相关推荐

  • C++设计模式编程中的迭代器模式应用解析

    迭代器模式:提供一种方法顺序访问一个聚合对象中个各个元素,而不暴露该对像的内部表示. 迭代器模式应该是最为熟悉的模式了,最简单的证明就是我在实现组合模式.享元模式.观察者模式中就直接用到了 STL 提供的迭代器来遍历 Vector 或者 List数据结构. 迭代器模式也正是用来解决对一个聚合对象的遍历问题,将对聚合的遍历封装到一个类中进行,这样就避免了暴露这个聚合对象的内部表示的可能. 模式的动机: (1)一个聚合对象,如一个列表(List)或者一个集合(Set),应该提供一种方法来让别人可以访

  • C++设计模式之迭代器模式(Iterator)

    迭代器在STL运用广泛,类似容器的迭代已经成为其重要特性,而迭代器模式则是利用迭代器概念进行的抽象运用,迭代器模式运用广泛和有用,因为其能够不考虑数据的存储方式,而是直接面对数据进行迭代,也就是说我们不用考虑集合是数组(或vector).链表.栈还是队列,而是通过统一的接口进行顺序的访问. 作用 迭代器模式提供了一种顺序访问容器中元素的方法,而无需了解器内部的类型和结构,该模式的核心思想将访问和遍历容器对象的功能交给一个外部的迭代器对象,该迭代器定义了访问聚合对象的接口, 类视图 实现 clas

  • C++ 模拟实现list(迭代器)实现代码

    C++ 模拟实现list(迭代器) 实现代码: #pragma once; #include <assert.h> #include<iostream> #include <assert.h> using namespace std; template<class T> struct __ListNode { T _data; __ListNode<T>* _next; __ListNode<T>* _prev; __ListNode

  • C++设计模式之迭代器模式

    前言 又到年底了,时间真的过的好快啊.最近也非常感伤,总是怀念大学的日子,做梦的时候也常常梦到.梦到大学在电脑前傻傻的敲着键盘,写着代码,对付着数据结构与算法的作业:建立一个链表,遍历链表,打印链表.现在把那个时候声明的链表的头文件拿出来看看: 复制代码 代码如下: typedef struct tagNode {      int value;      tagNode *pPre;      tagNode *pNext; }Node;   class CList { public:    

  • 详解C++中的vector容器及用迭代器访问vector的方法

    vector vector是相同类型对象的集合.集合中的每个对象有个对应的索引.vector常被称为容器(container). 为了使用vector,需要: #include <vector> using std::vector; vector是一个类模版(class template).C++有函数模版和类模版.模版本身不是函数或类,必须通过指定 类型让编译器去实例化(instantiation)它.比如vector<int> ivec. vector是模版,不是类型.从vec

  • C++begin和end运算符的返回迭代器的类型如何判断?

    begin和end返回的具体类型应该由对象是否是常量进行确定,如果对象是常量,则这两个函数返回const_iterator; 如果对象不是常量,则这个函数返回iterator类型.下面利用一个超级简单的小程序进行验证二者的类型,源代码如下: #include <iostream> #include <vector> using namespace std; int main() { vector<int> ivec; const vector<int> cv

  • 浅谈c++ stl迭代器失效的问题

    之前看<C++ Primier>的时候,也解到在顺序型窗口里insert/erase会涉及到迭代器失效的问题,并没有深究.今天写程序的时候遇到了这个问题. 1 莫名其妙的Erase 最初我的程序是酱紫的,别说话,我知道这样是有问题的,可这样是最直观的想法 int arr[]={0,1,2,3,4,5,6,7,8,9,10}; vector<int> a(arr,arr+sizeof(arr)/sizeof(*arr));for (auto it = a.begin(); it !=

  • C++迭代器介绍(iterator、const_iterator、reverse_interator、const_reverse_interator)

    概念:C++的一种机制,用来遍历标准模板库容器中的元素,是一种"智能"指针 一.迭代器的特点 迭代器是一种智能指针,具有遍历复杂数据结构的能力 不同的容器有不一样的内部结构,因此会有一样的迭代器类型 迭代器定义后,并不属于某一实例容器对象,只要是属于该迭代器类型的容器类型都可用 迭代器的分类 C++的STL定义了5种迭代器 输入迭代器:提供了对其指向元素的只读操作以及前++和后++操作符 输出迭代器:提供了对其指向元素的写操作和++操作符 向前迭代器:具有++操作符 双向迭代器:既具有

  • c++迭代器失效的情况汇总

    一.序列式容器(数组式容器) 对于序列式容器(如vector,deque),序列式容器就是数组式容器,删除当前的iterator会使后面所有元素的iterator都失效.这是因为vetor,deque使用了连续分配的内存,删除一个元素导致后面所有的元素会向前移动一个位置.所以不能使用erase(iter++)的方式,还好erase方法可以返回下一个有效的iterator. for (iter = cont.begin(); iter != cont.end();) { (*it)->doSome

  • 关于STL的erase()陷阱-迭代器失效问题的总结

    下面材料整理自Internet&著作. STL中的容器按存储方式分为两类,一类是按以数组形式存储的容器(如:vector .deque):另一类是以不连续的节点形式存储的容器(如:list.set.map).在使用erase方法来删除元素时,需要注意一些问题. 1.list,set,map容器 在使用 list.set 或 map遍历删除某些元素时可以这样使用: 1.1 正确写法1 std::list< int> List; std::list< int>::iterato

  • 关于vector迭代器失效的几种情况总结

    在泛型编程还是STL的实际运用中,迭代器(iterator)无疑扮演者重要的角色.迭代器是一种类似于指针的对象(如可以内容提领,成员访问等),但他又不仅仅是一种普通的指针. 关于迭代器失效,我们可以看下面这个例子: #include<vector> #include<list> void PrintVector(const vector<int>& v) { vector<int>::const_iterator it = v.begin(); wh

  • MySQL索引失效的几种情况汇总

    一.索引不存储null值 更准确的说,单列索引不存储null值,复合索引不存储全为null的值.索引不能存储Null,所以对这列采用is null条件时,因为索引上根本 没Null值,不能利用到索引,只能全表扫描. 为什么索引列不能存Null值? 将索引列值进行建树,其中必然涉及到诸多的比较操作.Null值的特殊性就在于参与的运算大多取值为null. 这样的话,null值实际上是不能参与进建索引的过程.也就是说,null值不会像其他取值一样出现在索引树的叶子节点上. 二.不适合键值较少的列(重复

  • 关于Mysql5.7及8.0版本索引失效情况汇总

    目录 一个独立索引 多个独立索引 总结 TIPS: 没有特殊说明,测试环境均为MySQL8.0,早期版本可能会有更多情况导致索引失效.8.0失效的情况,早期版本也失效:8.0不失效的情况,早期版本可能失效. 所有测试默认不考虑表为空的情况,特殊情况文中会有说明. 本文只介绍Innodb引擎下的索引失效情况. -- 创建测试表 DROP TABLE IF EXISTS `test_idx`; CREATE TABLE `test_idx` ( `id` int(11) NOT NULL AUTO_

  • vector list map 遍历删除制定元素 防止迭代器失效的实例

    方法如下所示: // k_control.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include "stdio.h" #include <vector> #include <map> #include <string> #include <list> using namespace std; int _tmain(int argc, _TCHAR* argv[]) {

  • 基于list循环删除元素,迭代器失效的问题详解

    问题的关键是:在删除元素之前,将当前迭代器保存下来.当然,这里仅支持list,因为list的链式的删除一个元素,前面的指针指向下一个元素,vector和queue就不好办了,它们或者是线性的或者是半线性半链式,迭代器会失效 #include<iostream> #include<list> using namespace std; int main() { list<int *> l; for(int i=1;i<=100;i++) { int* temp=new

  • 详解Vue 数据更新了但页面没有更新的 7 种情况汇总及延伸总结

    如果你发现你自己需要在 Vue 中做一次强制更新,99.9% 的情况,是你在某个地方做错了事. 1. Vue 无法检测实例被创建时不存在于 data 中的 property 原因:由于 Vue 会在初始化实例时对 property 执行 getter/setter 转化,所以 property 必须在 data 对象上存在才能让 Vue 将它转换为响应式的. 场景: var vm = new Vue({ data:{}, // 页面不会变化 template: '<div>{{message}

  • C++中vector迭代器失效问题详解

    目录 问题: (1)删除vector中所有的偶数 (2)vector容器插入元素问题 迭代器失效原因 解决: 总结 问题: (1)删除vector中所有的偶数 #include <iostream> #include <vector> using namespace std; int main() { vector<int> vec; for (int i = 0; i < 10; ++i) { vec.push_back(i); } //把vec容器中的所有偶数

随机推荐