【C++私房菜】序列式容器的迭代器失效问题
目录
一、list的迭代器失效
二、vector的迭代器失效
1、空间缩小操作
2、空间扩大操作
三、总结
在C++中,当对容器进行插入或删除操作时,可能会导致迭代器失效的问题。所谓迭代器失效指的是,原先指向容器中某个元素的迭代器,在容器发生结构性变化(比如插入、删除元素)后,可能不再指向之前预期的位置,甚至变得无效,不能再安全地使用。
迭代器失效通常会导致程序出现未定义行为,比如访问无效内存地址、产生崩溃等问题。这是因为在容器发生结构性变化时,迭代器所持有的指针或引用可能已经不再有效,但程序仍然试图通过这些失效的迭代器来访问容器中的内容,从而导致错误。
本文别以list和vector为例,给出代码示例并分析迭代器失效的情况。
一、list的迭代器失效
此处大家可将迭代器暂时理解成类似于指针,迭代器失效即迭代器所指向的节点的无效,即该节点被删除了。因为list的底层结构为带头结点的双向循环链表,因此在list中进行插入时是不会导致list的迭代器失效的,只有在删除时才会失效,并且失效的只是指向被删除节点的迭代器,其他迭代器不会受到影响。下面我们来了解一下list的erase函数:

该函数用于从list容器中删除单个元素或者一个范围内的元素。删除元素会导致容器大小减少,并且被销毁。与其他标准序列容器不同,list和forward_list对象专门设计用于在任何位置高效地插入和删除元素,即使是在序列的中间位置。参数包括position(指向要从list中删除的单个元素的迭代器),以及[first, last)(指定要删除的范围的迭代器,包括first指向的元素但不包括last指向的元素)。返回值是一个迭代器,指向函数调用erase的最后一个元素之后的元素。如果操作erase了序列中的最后一个元素,则返回容器的末尾位置。迭代器类型iterator是一个双向迭代器类型,用于指向元素。
-
list的迭代器失效问题
先看一个正常使用迭代器的例子:
#include <iostream>#include <list>int main() {std::list<int> myList = {1, 2, 3, 4, 5};auto it = myList.begin();// 在迭代器指向位置2之后插入一个元素++it; // 移动到位置2myList.insert(it, 10);for (auto it = myList.begin(); it != myList.end(); ++it) {std::cout << *it << " ";}std::cout << std::endl;//1 10 2 3 4 5 return 0;}
在上面的代码中,我们在list中插入元素时,使用了insert方法来在迭代器指向的位置后面插入一个新的元素。这样做是安全的,因为insert方法会返回一个指向新插入元素的迭代器,原先的迭代器仍然有效。
接下来,再看一个list的迭代器失效问题:
#include <iostream>#include <list>int main() {std::list<int> myList = {1, 2, 3, 4, 5};auto it = myList.begin();// 删除迭代器指向的位置2处的元素++it; // 移动到位置2myList.erase(it);//cout << *it; for (auto it = myList.begin(); it != myList.end(); ++it) {std::cout << *it << " ";}std::cout << std::endl;return 0;}
在这个例子中,我们删除了迭代器指向的位置2处的元素。此时,原先指向位置2的迭代器已经失效,应该避免继续使用它。即erase()函数执行后,it所指向的节点已被删除,因此it无效,在下一次使用it时,必须先给其赋值。
如果我们在 myList.erase(it);后输入 cout << *it;,在vs下会报如下错误:

这个错误信息表明程序中出现了尝试对值初始化的list迭代器进行解引用的情况。当你试图对指向列表中无效元素的迭代器进行解引用时,会导致未定义的行为,并可能引发断言失败。
在调用 erase 函数之后,被删除元素的迭代器会失效,因此不能再安全地对它进行解引用操作。在这种情况下,尝试输出 *it 会导致错误,因为 it 已经不再指向有效的元素了。
要避免这个问题,我们可以在调用 erase 函数之后,更新你的迭代器,使其指向正确的位置,或者使用 it = myList.erase(it); 来获取指向下一个有效元素的迭代器。
我们要避免这个问题,应该始终在对迭代器进行解引用操作之前检查它是否有效。你可以将迭代器与 list.end() 进行比较,以确定它是否指向列表的末尾,然后再尝试访问它所指向的元素。
二、vector的迭代器失效
迭代器的主要作用就是让算法能够不用关心底层数据结构,其底层实际就是一个指针,或者是对指针进行了封装,比如:vector的迭代器就是原生态指针T* 。因此迭代器失效,实际就是迭代器底层对应指针所指向的 空间被销毁了,而使用一块已经被释放的空间,造成的后果是程序崩溃((即如果继续使用已经失效的迭代器, 程序可能会崩溃)。vector导致迭代器失效的情景是引起其底层空间改变的函数或操作。
在C++的STL中,vector容器可以动态地增长和收缩,以适应元素数量的变化。当向vector容器中插入元素时,如果元素数量超过了当前容器的大小,vector会进行空间扩展操作;而当从vector容器中删除元素时,如果元素数量变少到一定程度,vector可能会进行空间收缩操作。
我们从两个方面来谈:
1、空间缩小操作
当使用pop_back()函数删除元素,且元素数量减少到一定程度以下时,vector可能会执行空间收缩操作。具体的收缩条件和实现细节因编译器和STL库的不同而有所差异。一般来说,vector并不会在每次删除元素后立即收缩内存空间,而是在适当的时机进行调整以提高性能。
使用erase()函数删除元素,同样可能触发空间收缩操作。
下面我们来了解一下vector的erase函数,我们仅使用erase函数来描述空间缩小的情况:

该函数用于从vector中删除单个元素或者一个范围内的元素。删除元素会导致容器大小减少,并且被销毁。由于vector使用数组作为其底层存储,因此在除了末尾位置之外的位置上擦除元素会导致容器重新定位被擦除段之后的所有元素到它们的新位置。与其他类型的序列容器(如list或forward_list)相比,这通常是一种低效的操作。参数包括position(指向要从vector中删除的单个元素的迭代器)和[first, last)(指定要删除的范围的迭代器,包括first指向的元素但不包括last指向的元素)。返回值是一个迭代器,指向函数调用erase的最后一个元素之后的新位置。如果操作erase了序列中的最后一个元素,则返回容器的末尾位置。
#include <iostream>#include <vector>int main() {std::vector<int> myVector = {1, 2, 3, 4, 5};auto it = myVector.begin();// 删除迭代器指向的位置2处的元素it = it + 2; // 移动到位置2myVector.erase(it);for (auto it = myVector.begin(); it != myVector.end(); ++it) {std::cout << *it << " ";}std::cout << std::endl;return 0;}
在这个例子中,我们删除了迭代器指向的位置2处的元素。与list类似,删除操作后原先的迭代器已经失效,应该避免继续使用它。
再例如如下案例:
#include <iostream>#include <vector>using namespace std;int main(){vector<int> v{ 1, 2, 3, 4 };auto pos = v.begin();while (pos != v.end()){if (*pos % 2 == 0)v.erase(pos);++pos;}return 0;}
erase删除pos位置元素后,pos位置之后的元素会往前移,没有导致底层空间的改变,理论上讲迭代器不应该会失效,但是:如果pos刚好是最后一个元素,删完之后pos刚好是end的位置,而end位置是 没有元素的,那么pos就失效了。因此删除vector中任意位置上元素时,vs就认为该位置迭代器失效了。vs下直接报错:

根据上面的内容,我们应在删除元素后,对迭代器进行赋值,使操作合法化:
#include <iostream>#include <vector>using namespace std;int main(){vector<int> v{ 1, 2, 3, 4 };auto it = v.begin();while (it != v.end()){if (*it % 2 == 0)it = v.erase(it);else ++it;}return 0;}
2、空间扩大操作
当使用push_back()函数向vector末尾添加元素,并且当前元素数量已经达到了vector的容量上限时,vector会执行空间扩大操作。通常情况下,vector会重新分配更大的内存空间,将原有元素拷贝到新的内存空间中,并释放原来的内存空间。
使用insert()函数在任意位置插入元素,如果导致vector超出容量上限,也会触发空间扩大操作。
下面我们来了解一下vector的erase函数,我们仅使用erase函数来描述空间缩小的情况:
#include <iostream>#include <vector>int main() {std::vector<int> myVector = {1, 2, 3, 4, 5};auto it = myVector.begin();// 在迭代器指向位置2之后插入一个元素it = it + 2; // 移动到位置2myVector.insert(it, 10);for (auto it = myVector.begin(); it != myVector.end(); ++it) {std::cout << *it << " ";}std::cout << std::endl;return 0;}
我们在vector中插入元素时,使用了insert方法并通过迭代器移动到指定位置。以上操作可能会导致vector扩容,也就是说vector底层原理旧空间被释放掉, 而在打印时,如cout << *it;,it还使用的是释放之间的旧空间,在对it迭代器操作时,实际操作的是一块已经被释放的 空间,而引起代码运行时崩溃。
与vector类似,string在插入+扩容操作+erase之后,迭代器也会失效。本文不再赘述,请读者结合vector理解。
需要注意的是,不同的编译器有不同的处理方式,Linux下,g++编译器对迭代器失效的检测并不是非常严格,处理也没有vs下极端。扩容之后,迭代器已经失效了,程序虽然可以运行,但是运行结果已经不对。或者erase删除任意位置代码后,Linux下迭代器并没有失效,因为空间还是原来的空间,后序元素往前搬移了,it的位置还是有效的。
三、总结
因此,在实际编程中,当对list、vector以及string 进行插入或删除操作时,需要格外小心,避免在迭代器失效的情况下继续使用迭代器。如果需要在循环中对容器进行插入或删除操作,可以考虑使用迭代器的insert和erase方法,并注意更新迭代器的位置,以避免迭代器失效问题。
一句话就能总结解决迭代器失效问题:在使用前,对迭代器重新赋值即可。
为了避免迭代器失效问题,通常建议在对容器进行插入或删除操作时,谨慎处理迭代器的使用:
-
在循环中进行插入或删除操作时,可以考虑使用迭代器的insert和erase方法,这些方法会返回一个新的迭代器,避免原迭代器失效。
-
插入或删除元素后,及时更新迭代器的位置,确保迭代器指向的是正确的元素。
-
避免在迭代器失效的情况下继续使用迭代器。
总之,迭代器失效是指迭代器指向的位置在容器结构发生变化后不再有效,因此在对容器进行修改操作时,需要特别注意迭代器的使用,以避免出现迭代器失效导致的问题。
相关文章:
【C++私房菜】序列式容器的迭代器失效问题
目录 一、list的迭代器失效 二、vector的迭代器失效 1、空间缩小操作 2、空间扩大操作 三、总结 在C中,当对容器进行插入或删除操作时,可能会导致迭代器失效的问题。所谓迭代器失效指的是,原先指向容器中某个元素的迭代器,在…...
MySQL 篇-深入了解 DML、DQL 语言(二)
🔥博客主页: 【小扳_-CSDN博客】 ❤感谢大家点赞👍收藏⭐评论✍ 文章目录 1.0 DML、DQL 语言说明 2.0 使用 DML 实现对数据管理和操作 2.1 DML - 增添数据 insert 2.2 DML - 修改数据 update 2.3 DML - 删除数据 delete 3.0 使用 DQL 实现对…...
端智能:面向手机计算环境的端云协同AI技术创新
近年来,随着移动端设备软硬件能力的进步,移动端的算力有了很大提升,同时面向移动端的机器学习框架和模型轻量化技术越来越成熟,端上的AI能力逐渐进入大众视野,端智能在电商领域也开始逐步走向规模化应用。通过持续探索…...
PHP函数 “password_hash“ 哈希密码
哈希函数是一种将输入转换为固定长度字符串的方法,这个过程是不可逆的,也就是无法从哈希值还原出原始输入。通过将密码进行哈希处理,即使数据库泄露,攻击者也无法简单地获取到用户密码。 在PHP中,我们可以使用 "…...
第十一天-Excel的操作
目录 1.xlrd-Excel的读模块 安装 使用 获取工作簿 读取工作簿的内容 xlsxwriter-Excel的写模块 安装 使用 生成图表 add_series参数 图表的样式 demo:生成图表 Excel的操作在python中有多个模块,为了能够快速使用,选择了相对简单…...
【java任意文件漏洞修复,使用文件魔数解决】
java任意文件漏洞修复,使用文件魔数解决 背景: 客户进行渗透测试,验证上传文件的程序没有对上传文件作任何过滤,导致可以上传任意文件到服务器,甚至是病毒文件和Webshell木马文件。 解决办法:对于上传的附件…...
LeetCode 热题 100 | 二叉树(二)
目录 1 543. 二叉树的直径 2 102. 二叉树的层序遍历 3 108. 将有序数组转换为二叉搜索树 菜鸟做题,语言是 C 1 543. 二叉树的直径 这道题和 124. 二叉树中的最大路径和 太像了 题眼:二叉树的 直径 是指树中任意两个节点之间 最长路径的长度 。…...
mini-spring|定义标记类型Aware接口,实现感知容器对象
**前言:**如果我们想获得 Spring 框架提供的 BeanFactory、ApplicationContext、BeanClassLoader等这些能力做一些扩展框架的使用时该怎么操作呢。所以我们本章节希望在 Spring 框架中提供一种能感知容器操作的接口,如果谁实现了这样的一个接口ÿ…...
83. 删除排序链表中的重复元素
给定一个已排序的链表的头 head , 删除所有重复的元素,使每个元素只出现一次 。返回 已排序的链表 。 输入:head = [1,1,2] 输出:[1,2] 输入:head = [1,1,2,3,3] 输出:[1,2,3] 提示: 链表中节点数目在范围 [0, 300] 内-100 <= Node.val <= 100题目数据保证链表已…...
贪心算法
贪心算法 例题1、股票买卖题目信息思路题解 2、货仓选址题目信息思路题解 3、糖果传递题目信息思路题解 4、雷达设备题目信息思路题解 例题 1、股票买卖 题目信息 思路 相邻两天,后>前,则交易一次 题解 #include <bits/stdc.h> #define en…...
MySQL基本知识
目录 一,MySQL的元数据库 1.1.什么是元数据库 1.2.有哪些元数据库 1.3.切换数据库 二,账户管理 2.1.设置权限 2.2.授权用户 2.3.查看权限 2.4.撤销权限 三,MySQL引擎 3.1什么是数据库引擎 3.2.查看数据引擎 3.3.MyISAM引擎 3.4…...
Vue3 (unplugin-auto-import自动导入的使用)
安装 参考链接 npm i -D unplugin-auto-importvite.config.ts里面配置 import AutoImport from unplugin-auto-import/viteAutoImport({imports:[ vue,vue-router]})重新运行项目会生成一个auto-imports.d.ts的文件 /* eslint-disable */ /* prettier-ignore */ // ts-nochec…...
【漏洞复现】大华智慧园区综合管理平台信息泄露漏洞
Nx01 产品简介 大华智慧园区综合管理平台是一款综合管理平台,具备园区运营、资源调配和智能服务等功能。该平台旨在协助优化园区资源分配,满足多元化的管理需求,同时通过提供智能服务,增强使用体验。 Nx02 漏洞描述 大华智慧园区…...
JavaScript的书写方式
JavaScript的书写方式 目前较为流行的是第二种和第三种,第一种很少见。在第二种和第三种推荐使用第三种,因为在日常开发/工作中,第三种是最为常见的 1.行内式 把JS代码嵌入到html元素内部 示例代码 运行效果 由于JS中字符串常量可以使用单引…...
第二十篇-推荐-纯CPU(E5-2680)推理-llama.cpp-qwen1_5-72b-chat-q4_k_m.gguf
环境 系统:CentOS-7 CPU: Intel Xeon CPU E5-2680 v4 2.40GHz 14C28T 内存: 48G DDR3 依赖安装 make --version GNU Make 4.3gcc --version gcc (GCC) 11.2.1 20220127 (Red Hat 11.2.1-9)g --version g (GCC) 11.2.1 20220127 (Red Hat …...
CSS常见选择器
CSS常见选择器 在Web开发中,层叠样式表(CSS)是用于描述HTML或XML(包括SVG和XHTML等其他XML语言)文档的样式的语言。CSS描述了文档的表现形式,包括布局、颜色和字体等。在CSS中,选择器是一种模式…...
[LWC] Components Communication
目录 Overview Summary Sample Code 1. Parent -> Child - Public Setter / Property / Function a. Public Property b. Public getters and setters c. Public Methods 2. Child -> Parent - Custom Event 3. Unrelated Components - LMS (Lightning Message…...
Unity中URP实现水体(水下的扭曲)
文章目录 前言一、使用一张法线纹理,作为水下扭曲的纹理1、在属性面板定义一个纹理,用于传入法线贴图2、在Pass中,定义对应的纹理和采样器3、在常量缓冲区,申明修改 Tilling 和 Offset 的ST4、在顶点着色器,计算得到 应…...
anaconda指定目录创建环境无效/环境无法创建到指定位置
已经设置目录到D盘 创建环境时还是分配到C盘 可能是指定位置没有开启读写权限,如我在这里安装到了anaconda文件夹,则打开该文件夹的属性->安全->编辑 allusers下的权限全都打勾...
《Docker极简教程》--Docker在生产环境的应用--Docker在生产环境的部署
一、准备工作 1.1 硬件和基础设施要求 硬件和基础设施要求是在部署 Docker 到生产环境之前需要认真考虑和准备的重要方面,以下是一般性的要求: 服务器硬件: CPU:建议使用多核处理器,以支持同时运行多个容器。内存&a…...
使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式
一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明:假设每台服务器已…...
华为云AI开发平台ModelArts
华为云ModelArts:重塑AI开发流程的“智能引擎”与“创新加速器”! 在人工智能浪潮席卷全球的2025年,企业拥抱AI的意愿空前高涨,但技术门槛高、流程复杂、资源投入巨大的现实,却让许多创新构想止步于实验室。数据科学家…...
OpenLayers 可视化之热力图
注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 热力图(Heatmap)又叫热点图,是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...
【Linux】shell脚本忽略错误继续执行
在 shell 脚本中,可以使用 set -e 命令来设置脚本在遇到错误时退出执行。如果你希望脚本忽略错误并继续执行,可以在脚本开头添加 set e 命令来取消该设置。 举例1 #!/bin/bash# 取消 set -e 的设置 set e# 执行命令,并忽略错误 rm somefile…...
51c自动驾驶~合集58
我自己的原文哦~ https://blog.51cto.com/whaosoft/13967107 #CCA-Attention 全局池化局部保留,CCA-Attention为LLM长文本建模带来突破性进展 琶洲实验室、华南理工大学联合推出关键上下文感知注意力机制(CCA-Attention),…...
shell脚本--常见案例
1、自动备份文件或目录 2、批量重命名文件 3、查找并删除指定名称的文件: 4、批量删除文件 5、查找并替换文件内容 6、批量创建文件 7、创建文件夹并移动文件 8、在文件夹中查找文件...
聊聊 Pulsar:Producer 源码解析
一、前言 Apache Pulsar 是一个企业级的开源分布式消息传递平台,以其高性能、可扩展性和存储计算分离架构在消息队列和流处理领域独树一帜。在 Pulsar 的核心架构中,Producer(生产者) 是连接客户端应用与消息队列的第一步。生产者…...
系统设计 --- MongoDB亿级数据查询优化策略
系统设计 --- MongoDB亿级数据查询分表策略 背景Solution --- 分表 背景 使用audit log实现Audi Trail功能 Audit Trail范围: 六个月数据量: 每秒5-7条audi log,共计7千万 – 1亿条数据需要实现全文检索按照时间倒序因为license问题,不能使用ELK只能使用…...
【python异步多线程】异步多线程爬虫代码示例
claude生成的python多线程、异步代码示例,模拟20个网页的爬取,每个网页假设要0.5-2秒完成。 代码 Python多线程爬虫教程 核心概念 多线程:允许程序同时执行多个任务,提高IO密集型任务(如网络请求)的效率…...
【JavaSE】绘图与事件入门学习笔记
-Java绘图坐标体系 坐标体系-介绍 坐标原点位于左上角,以像素为单位。 在Java坐标系中,第一个是x坐标,表示当前位置为水平方向,距离坐标原点x个像素;第二个是y坐标,表示当前位置为垂直方向,距离坐标原点y个像素。 坐标体系-像素 …...
