【C++私房菜】序列式容器的迭代器失效问题

目录

一、list的迭代器失效

二、vector的迭代器失效

1、空间缩小操作

2、空间扩大操作

三、总结


在C++中,当对容器进行插入或删除操作时,可能会导致迭代器失效的问题。所谓迭代器失效指的是,原先指向容器中某个元素的迭代器,在容器发生结构性变化(比如插入、删除元素)后,可能不再指向之前预期的位置,甚至变得无效,不能再安全地使用。

迭代器失效通常会导致程序出现未定义行为,比如访问无效内存地址、产生崩溃等问题。这是因为在容器发生结构性变化时,迭代器所持有的指针或引用可能已经不再有效,但程序仍然试图通过这些失效的迭代器来访问容器中的内容,从而导致错误。

本文别以list和vector为例,给出代码示例并分析迭代器失效的情况。


一、list的迭代器失效

此处大家可将迭代器暂时理解成类似于指针,迭代器失效即迭代器所指向的节点的无效,即该节点被删除了。因为list的底层结构为带头结点的双向循环链表,因此在list中进行插入时是不会导致list的迭代器失效的,只有在删除时才会失效,并且失效的只是指向被删除节点的迭代器,其他迭代器不会受到影响。下面我们来了解一下listerase函数:

该函数用于从list容器中删除单个元素或者一个范围内的元素。删除元素会导致容器大小减少,并且被销毁。与其他标准序列容器不同,listforward_list对象专门设计用于在任何位置高效地插入和删除元素,即使是在序列的中间位置。参数包括position(指向要从list中删除的单个元素的迭代器),以及[first, last)(指定要删除的范围的迭代器,包括first指向的元素但不包括last指向的元素)。返回值是一个迭代器,指向函数调用erase的最后一个元素之后的元素。如果操作erase了序列中的最后一个元素,则返回容器的末尾位置。迭代器类型iterator是一个双向迭代器类型,用于指向元素。

  1. 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()函数删除元素,同样可能触发空间收缩操作。

下面我们来了解一下vectorerase函数,我们仅使用erase函数来描述空间缩小的情况:

该函数用于从vector中删除单个元素或者一个范围内的元素。删除元素会导致容器大小减少,并且被销毁。由于vector使用数组作为其底层存储,因此在除了末尾位置之外的位置上擦除元素会导致容器重新定位被擦除段之后的所有元素到它们的新位置。与其他类型的序列容器(如listforward_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超出容量上限,也会触发空间扩大操作。

下面我们来了解一下vectorerase函数,我们仅使用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的位置还是有效的。


三、总结

因此,在实际编程中,当对listvector以及string 进行插入或删除操作时,需要格外小心,避免在迭代器失效的情况下继续使用迭代器。如果需要在循环中对容器进行插入或删除操作,可以考虑使用迭代器的insert和erase方法,并注意更新迭代器的位置,以避免迭代器失效问题。

一句话就能总结解决迭代器失效问题:在使用前,对迭代器重新赋值即可。

为了避免迭代器失效问题,通常建议在对容器进行插入或删除操作时,谨慎处理迭代器的使用:

  • 在循环中进行插入或删除操作时,可以考虑使用迭代器的insert和erase方法,这些方法会返回一个新的迭代器,避免原迭代器失效。

  • 插入或删除元素后,及时更新迭代器的位置,确保迭代器指向的是正确的元素。

  • 避免在迭代器失效的情况下继续使用迭代器。

总之,迭代器失效是指迭代器指向的位置在容器结构发生变化后不再有效,因此在对容器进行修改操作时,需要特别注意迭代器的使用,以避免出现迭代器失效导致的问题。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/2815224.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

STM32_DS18B20_1_芯片简介及初始化配置

DS18B20介绍 DS18B20数字温度计提供9位到12位摄氏度的温度测量&#xff0c;并具有非易失性&#xff0c;用户可编程的上下触发点的报警功能。DS18B20通过1线总线进行通信&#xff0c;根据定义&#xff0c;该总线只需要一条数据线&#xff0c;即可与中央微处理器进行通信…

5G双域快网

目录 一、业务场景 二、三类技术方案 2.1、专用DNN方案 2.2、ULCL方案&#xff1a;通用/专用DNNULCL分流 2.3、 多DNN方案-定制终端无感分流方案 漫游场景 一、业务场景 初期双域专网业务可划分为三类业务场景&#xff0c;学校、政务、文旅等行业均已提出公/专网融合访问需…

每日五道java面试题之spring篇(九)

目录&#xff1a; 第一题. 说一下Spring的事务传播行为第二题. 说一下 spring 的事务隔离&#xff1f;第三题. Spring AOP and AspectJ AOP 有什么区别&#xff1f;AOP 有哪些实现方式&#xff1f;第四题. JDK动态代理和CGLIB动态代理的区别第五题. 解释一下Spring AOP里面的几…

nginx实现http反向代理及负载均衡

目录 一、代理概述 1、代理概念 1.1 正向代理&#xff08;Forward Proxy&#xff09; 1.2 反向代理&#xff08;Reverse Proxy&#xff09; 1.3 正向代理与反向代理的区别 2、同构代理与异构代理 2.1 同构代理 2.2 异构代理 2.3 同构代理与异构代理的区别 二、四层代…

VL817-Q7 USB3.0 HUB芯片 适用于扩展坞 工控机 显示器

VL817-Q7 USB3.1 GEN1 HUB芯片 VL817-Q7 USB3.1 GEN1 HUB芯片 VIA Lab的VL817是一款现代USB 3.1 Gen 1集线器控制器&#xff0c;具有优化的成本结构和完全符合USB标准3.1 Gen 1规范&#xff0c;包括ecn和2017年1月的合规性测试更新。VL817提供双端口和双端口4端口配置&…

Linux NFC 子系统剖析

1.总览 linux源码中NFC在net/nfc下&#xff0c;文件结构如下图&#xff1a; hci&#xff1a;Host Controller Interface 主要是针对NFC的主机-控制器接口协议 nci&#xff1a;NFC Controller Interface 主要是NFC的控制器接口协议&#xff0c;用于NFCC(NFC Controller)和DH(…

【Go语言】Go语言中的切片

Go语言中的切片 1.切片的定义 Go语言中&#xff0c;切片是一个新的数据类型数据类型&#xff0c;与数组最大的区别在于&#xff0c;切片的类型中只有数据元素的类型&#xff0c;而没有长度&#xff1a; var slice []string []string{"a", "b", "c…

GCC的符号可见性: 解决Linux多个库同名符号冲突问题以及引用不同版本库的问题

目录 1 -fvisibilitydefault|internal|hidden|protected 1.1 __attribute__((visibility("default"))) 与 CXXg -fvisibilityhidden 的作用 1.2 __attribute__((visibility("hidden"))) 与 CXXg -fvisibilitydefault的作用 2 我的问题 2.1 解决措…

雾锁王国服务器怎么建?雾锁王国服务器搭建方法

雾锁王国Enshrouded服务器搭建怎么搭建&#xff1f;非常简单&#xff0c;阿里云计算巢雾锁王国程序&#xff0c;可以一键搭建雾锁王国多人联机服务器&#xff0c;腾讯云是基于雾锁王国镜像系统&#xff0c;阿里云服务网aliyunfuwuqi.com汇总雾锁王国服务器搭建&#xff0c;超简…

kafka三节点集群平滑升级过程指导

一、前言 Apache Kafka作为常用的开源分布式流媒体平台&#xff0c;可以实时发布、订阅、存储和处理数据流,多用于作为消息队列获取实时数据&#xff0c;构建对数据流的变化进行实时反应的应用程序&#xff0c;已被数千家公司用于高性能数据管道、流分析、数据集成和任务关键型…

DolphinScheduler——工作流实例的生命周期

目录 一、DolphinScheduler架构原理 1.1 系统架构图 1.2 DolphinScheduler核心概念 1.2 创建工作流 1.2.1 如何触发一个工作流实例 1.2.2 任务调度链路监控 1.2.3 Workflow-DAG解析 DAG解析 Dispatch分发流程 Master和Worker的交互过程 1.3 任务运行状态 该篇文章主…

就业班 2401--2.28 Linux Day7--存储管理1

一 .存储管理 主要知识点: 基本分区、逻辑卷LVM、EXT3/4/XFS文件系统、RAID 初识硬盘 机械 HDD 固态 SSD SSD的优势 SSD采用电子存储介质进行数据存储和读取的一种技术&#xff0c;拥有极高的存储性能&#xff0c;被认为是存储技术发展的未来新星。 与传统硬盘相比&#…

Codeforces Round 929 (Div. 3)(A,B,C,D,E,F,G)

这场没考什么算法&#xff0c;比较水&#xff0c;难度也不是很高。比赛链接 硬要说的话E有个 前缀和 加 二分&#xff0c;F是数学BFS&#xff0c;G是个构造 A. Turtle Puzzle: Rearrange and Negate 题意&#xff1a; 给你一个由 n n n 个整数组成的数组 a a a 。您必须对…

Rocky Linux 运维工具yum

一、yum的简介 ​​yum​是用于在基于RPM包管理系统的包管理工具。用户可以通过 ​yum​来搜索、安装、更新和删除软件包&#xff0c;自动处理依赖关系&#xff0c;方便快捷地管理系统上的软件。 二、yum的参数说明 1、install 用于在系统的上安装一个或多个软件包 2、seach 用…

golang使用gorm操作mysql1

1.mysql连接配置 package daoimport ("fmt""gorm.io/driver/mysql""gorm.io/gorm""gorm.io/gorm/logger" )var DB *gorm.DB// 连接数据库&#xff0c;启动服务的时候&#xff0c;init方法就会执行 func init() {username : "roo…

stm32学习笔记:ADC

1 ADC简介 ADC的作用ADC就是一个电压表&#xff0c;把引脚的电压值测出来&#xff0c;放在一个变量里 DAC的作用信号发生器、音频解码芯片 ADC的两个关键参数&#xff1a; 1、分辨率&#xff0c;一般用多少位来表示&#xff0c;12位AD值&#xff0c;它的表示范围就是0~2^12-1&…

Modern C++ std::any为何要求Tp可拷贝构造?

小问题也会影响设计的思路&#xff0c;某个问题或某种case的探讨有助于理解设计的初衷。 声明&#xff1a;以下_Tp/Tp都是指要放入std::any的对象的类型。 它要求_Tp is_copy_constructible, 仅仅是因为有很多函数的实现调用了Tp的拷贝构造函数吗&#xff1f;比如说上节提到的初…

pikachu之xss获取键盘记录

前备知识 跨域 跨域&#xff08;Cross-Origin&#xff09;是指在互联网中&#xff0c;浏览器为了保护用户信息安全而实施的一种安全策略——同源策略&#xff08;Same-Origin Policy&#xff09;&#xff0c;即浏览器禁止一个域上的文档或者脚本&#xff08;如通过JavaScript发…

Rocky Linux安装部署Elasticsearch(ELK日志服务器)

一、Elasticsearch的简介 Elasticsearch是一个强大的开源搜索和分析引擎&#xff0c;可用于实时处理和查询大量数据。它具有高性能、可扩展性和分布式特性&#xff0c;支持全文搜索、聚合分析、地理空间搜索等功能&#xff0c;是构建实时应用和大规模数据分析平台的首选工具。 …

(十七)devops持续集成开发——使用jenkins流水线pipeline方式发布一个微服务项目

前言 本节内容是一套关于微服务项目在docker环境中使用jenkins流水线部署的完整方案,在开始本节内容之前,我们需要提前安装好docker环境,以及docker本地镜像仓库docker harbor,同时安装好SonarQube用于代码验证,具体的安装步骤可参考作者的往期博客内容。 正文 在源码仓…