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

发布时间 - 2026-01-10 22:05:30    点击率:

下面材料整理自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>::iterator itList;
for( itList = List.begin(); itList != List.end(); )
{
   if( WillDelete( *itList) )
   {
      itList = List.erase( itList);
    }
    else
      itList++;
}

1.2 正确写法2

std::list< int> List;
std::list< int>::iterator itList;
for( itList = List.begin(); itList != List.end(); )
{
   if( WillDelete( *itList) )
   {
     List.erase( itList++);
   }
   else
     itList++;
}

1.3 错误写法1

std::list< int> List;
std::list< int>::iterator itList;
for( itList = List.begin(); itList != List.end(); itList++)
{
  if( WillDelete( *itList) )
  {
     List.erase( itList);
  }
}

1.4 错误写法2

std::list< int> List;
std::list< int>::iterator itList;
for( itList = List.begin(); itList != List.end(); )
{
   if( WillDelete( *itList) )
   {
     itList = List.erase( ++itList);
   }
   else
     itList++;
}

1.5 分析

正确使用方法1:通过erase方法的返回值来获取下一个元素的位置

正确使用方法2:在调用erase方法之前先使用 “++”来获取下一个元素的位置

错误使用方法1:在调用erase方法之后使用“++”来获取下一个元素的位置,由于在调用erase方法以后,该元素的位置已经

被删除,如果在根据这个旧的位置来获取下一个位置,则会出现异常。

错误使用方法2:同上。

2. vector,deque容器

在使用 vector、deque遍历删除元素时,也可以通过erase的返回值来获取下一个元素的位置:

2.1 正确写法

std::vector< int> Vec;
std::vector< int>::iterator itVec;
for( itVec = Vec.begin(); itVec != Vec.end(); )
{
 if( WillDelete( *itVec) )
 {
   itVec = Vec.erase( itVec);
  }
 else
   itList++;
}

2.2 注意

注意:vector、deque 不能像上面的“正确使用方法2”的办法来遍历删除。原因请参考Effective STL条款9。摘录到下面:

1)  对于关联容器(如map, set, multimap,multiset),删除当前的iterator,仅仅会使当前的iterator失效,只要在erase时,递增当前iterator即可。这是因为map之类的容器,使用了红黑树来实现,插入、删除一个结点不会对其他结点造成影响。

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

因为iter传给erase方法的是一个副本,iter++会指向下一个元素。

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

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

3)对于list来说,它使用了不连续分配的内存,并且它的erase方法也会返回下一个有效的iterator,因此上面两种方法都可以使用。

3.迭代器失效的情况

3.1 vector

内部数据结构:数组。

随机访问每个元素,所需要的时间为常量。

在末尾增加或删除元素所需时间与元素数目无关,在中间或开头增加或删除元素所需时间随元素数目呈线性变化。

可动态增加或减少元素,内存管理自动完成,但程序员可以使用reserve()成员函数来管理内存。

vector的迭代器在内存重新分配时将失效(它所指向的元素在该操作的前后不再相同)。当把超过capacity()-size()个元素插入vector中时,内存会重新分配,所有的迭代器都将失效;否则,指向当前元素以后的任何元素的迭代器都将失效。当删除元素时,指向被删除元素以后的任何元素的迭代器都将失效。

3.2 deque

内部数据结构:数组。

随机访问每个元素,所需要的时间为常量。

在开头和末尾增加元素所需时间与元素数目无关,在中间增加或删除元素所需时间随元素数目呈线性变化。

可动态增加或减少元素,内存管理自动完成,不提供用于内存管理的成员函数。

增加任何元素都将使deque的迭代器失效。在deque的中间删除元素将使迭代器失效。在deque的头或尾删除元素时,只有指向该元素的迭代器失效。

3.3 list

内部数据结构:双向环状链表。

不能随机访问一个元素。

可双向遍历。

在开头、末尾和中间任何地方增加或删除元素所需时间都为常量。

可动态增加或减少元素,内存管理自动完成。

增加任何元素都不会使迭代器失效。删除元素时,除了指向当前被删除元素的迭代器外,其它迭代器都不会失效。

3.4 slist

内部数据结构:单向链表。

不可双向遍历,只能从前到后地遍历。

其它的特性同list相似。

3.5 stack

适配器,它可以将任意类型的序列容器转换为一个堆栈,一般使用deque作为支持的序列容器。

元素只能后进先出(LIFO)。

不能遍历整个stack。

3.6 queue

适配器,它可以将任意类型的序列容器转换为一个队列,一般使用deque作为支持的序列容器。

元素只能先进先出(FIFO)。

不能遍历整个queue。

3.7 priority_queue

适配器,它可以将任意类型的序列容器转换为一个优先级队列,一般使用vector作为底层存储方式。

只能访问第一个元素,不能遍历整个priority_queue。

第一个元素始终是优先级最高的一个元素。

3.8 set

键和值相等。

键唯一。

元素默认按升序排列。

如果迭代器所指向的元素被删除,则该迭代器失效。其它任何增加、删除元素的操作都不会使迭代器失效。

3.9 multiset

键可以不唯一。

其它特点与set相同。

3.10 hash_set

与set相比较,它里面的元素不一定是经过排序的,而是按照所用的hash函数分派的,它能提供更快的搜索速度(当然跟hash函数有关)。

其它特点与set相同。

3.11 hash_multiset

键可以不唯一。

其它特点与hash_set相同。

3.12 map

键唯一。

元素默认按键的升序排列。

如果迭代器所指向的元素被删除,则该迭代器失效。其它任何增加、删除元素的操作都不会使迭代器失效。

3.13 multimap

键可以不唯一。

其它特点与map相同。

3.14 hash_map

与map相比较,它里面的元素不一定是按键值排序的,而是按照所用的hash函数分派的,它能提供更快的搜索速度(当然也跟hash函数有关)。

其它特点与map相同。

3.15 hash_multimap

键可以不唯一。

其它特点与hash_map相同。

以上就是小编为大家带来的关于STL的erase()陷阱-迭代器失效问题的总结全部内容了,希望大家多多支持~


# stl  # erase  # 迭代器失效  # 基于list循环删除元素  # 迭代器失效的问题详解  # 浅谈c++ stl迭代器失效的问题  # 迭代  # 遍历  # 所需  # 会使  # 数据结构  # 或删除  # 都不  # 都将  # 它可以  # 内存管理  # 转换为  # 升序  # 第一个  # 自动完成  # 时间为  # 更快  # 可以使用  # 它能  # 这是因为  # 所需要 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel模型关联查询教程_Laravel Eloquent一对多关联写法  Laravel如何实现本地化和多语言支持_Laravel多语言配置与翻译文件管理  非常酷的网站设计制作软件,酷培ai教育官方网站?  VIVO手机上del键无效OnKeyListener不响应的原因及解决方法  Laravel怎么实现验证码(Captcha)功能  教你用AI将一段旋律扩展成一首完整的曲子  韩国服务器如何优化跨境访问实现高效连接?  如何打造高效商业网站?建站目的决定转化率  JS去除重复并统计数量的实现方法  Laravel如何处理异常和错误?(Handler示例)  Edge浏览器怎么启用睡眠标签页_节省电脑内存占用优化技巧  Laravel请求验证怎么写_Laravel Validator自定义表单验证规则教程  如何在 Pandas 中基于一列条件计算另一列的分组均值  LinuxShell函数封装方法_脚本复用设计思路【教程】  夸克浏览器网页跳转延迟怎么办 夸克浏览器跳转优化  如何在腾讯云免费申请建站?  Laravel Octane如何提升性能_使用Laravel Octane加速你的应用  Win11怎么修改DNS服务器 Win11设置DNS加速网络【指南】  Win11搜索栏无法输入_解决Win11开始菜单搜索没反应问题【技巧】  利用JavaScript实现拖拽改变元素大小  如何在Windows 2008云服务器安全搭建网站?  Laravel如何使用Service Provider注册服务_Laravel服务提供者配置与加载  如何快速生成可下载的建站源码工具?  Firefox Developer Edition开发者版本入口  浅谈Javascript中的Label语句  Laravel怎么设置路由分组Prefix_Laravel多级路由嵌套与命名空间隔离【步骤】  Win11任务栏卡死怎么办 Windows11任务栏无反应解决方法【教程】  JavaScript如何操作视频_媒体API怎么控制播放  大同网页,大同瑞慈医院官网?  专业型网站制作公司有哪些,我设计专业的,谁给推荐几个设计师兼职类的网站?  动图在线制作网站有哪些,滑动动图图集怎么做?  Laravel如何实现多级无限分类_Laravel递归模型关联与树状数据输出【方法】  C#如何调用原生C++ COM对象详解  深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?  Laravel如何生成URL和重定向?(路由助手函数)  Laravel如何使用Spatie Media Library_Laravel图片上传管理与缩略图生成【步骤】  如何在HTML表单中获取用户输入并结合JavaScript动态控制复利计算循环  微信小程序 闭包写法详细介绍  如何在阿里云虚拟主机上快速搭建个人网站?  php静态变量怎么调试_php静态变量作用域调试技巧【解答】  Java遍历集合的三种方式  Python文件流缓冲机制_IO性能解析【教程】  如何在 Python 中将列表项按字母顺序编号(a.、b.、c. …)  如何在IIS7中新建站点?详细步骤解析  北京网站制作的公司有哪些,北京白云观官方网站?  创业网站制作流程,创业网站可靠吗?  Laravel怎么实现一对多关联查询_Laravel Eloquent模型关系定义与预加载【实战】  linux写shell需要注意的问题(必看)  如何在宝塔面板创建新站点?  如何在云主机上快速搭建网站?