【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; // 移动到位置2
     myList.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; // 移动到位置2
     myList.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; // 移动到位置2
     myVector.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; // 移动到位置2
     myVector.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方法,这些方法会返回一个新的迭代器,避免原迭代器失效。

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

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

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

IDEA基础——Maven配置tomcat

配置方案 一、配置maven-tomcat plugin插件&#xff08;只最高支持到tomcat 8&#xff09;~~1.添加镜像源&#xff0c;获取tomcat 8插件配置~~~~1.1 在pom.xml里先添加镜像源~~~~1.2 添加tomcat插件配置~~ 2. 添加tomact官方发布的插件配置&#xff08;无需添加镜像源&#xff…

回溯算法,你“回”了吗

目录 一、什么是回溯算法 二、应用场景 三、一般解题步骤 1、确定回溯方法以及参数 2、确定回溯的终止条件 3、确定搜索过程 四、力扣例题 1、题目描述 2、解题思路 3、代码示例 五、总结 一、什么是回溯算法 回溯算法&#xff0c;又称为试探法&#xff0c;是一种…

用友 NC 23处接口XML实体注入漏洞复现

0x01 产品简介 用友 NC 是用友网络科技股份有限公司开发的一款大型企业数字化平台。 0x02 漏洞概述 用友 NC 多处接口存在XML实体注入漏洞,未经身份验证攻击者可通过该漏洞读取系统重要文件(如数据库配置文件、系统配置文件)、数据库配置文件等等,导致网站处于极度不安全…

【Redis】深入理解 Redis 常用数据类型源码及底层实现(5.详解List数据结构)

本文是深入理解 Redis 常用数据类型源码及底层实现系列的第5篇&#xff5e;前4篇可移步(&#xffe3;∇&#xffe3;)/ 【Redis】深入理解 Redis 常用数据类型源码及底层实现&#xff08;1.结构与源码概述&#xff09;-CSDN博客 【Redis】深入理解 Redis 常用数据类型源码及底…

Ubuntu22.04.3LTS源码编译安装ffmpeg6.x

1.官网ffmpeg下载源码 https://ffmpeg.org/download.html#build-windows 安装 libx264 开发库&#xff08;一个开源的视频压缩库&#xff0c;用于编码视频流为 H.264/MPEG-4 AVC 视频格式&#xff09;。这是编译 FFmpeg 时如果要支持 H.264 编码必须的。 sudo apt install l…

Liunx前后端项目部署(小白也可安装)

文章目录 一、CentOS服务器的安装二、jdk安装三、Tomcat安装四、MySQL安装、五、nginX安装六、多个项目负载均衡&#xff0c;部署后端项目七、前端项目部署 一、CentOS服务器的安装 选择liunx&#xff0c;下面选择CentOS 7 ![在这里插入图片描述](https://img-blog.csdnimg.cn…

预训练概念

预训练是指在特定任务之前&#xff0c;在大规模数据集上对神经网络进行训练以学习通用的表示形式或特征。这些通用表示可以捕捉数据中的统计结构和语义信息&#xff0c;使得神经网络能够更好地理解和处理输入数据。 在大规模预训练模型中&#xff0c;通常会使用无监督或弱监督的…

python脚本实现全景站点矩阵转欧拉角

效果 脚本 import re import numpy as np import math import csv from settings import * # 以下是一个示例代码,可以输入3*3旋转矩阵,然后输出旋转角度:# ,输入3*3旋转矩阵# 计算x,y,z旋转角def rotation_matrix_to_euler_angles(R):

JVM(2)

JVM类加载 指的是java进程运行时,需要把.class文件从硬盘加载到内存,并进行一系列校验解析的过程. 核心: .class文件>类对象; 硬盘>内存. 类加载过程 在整个JVM的执行流程中,和程序员关系最密切的就是类加载的过程了,所以我们来看一下类加载的执行流程. 对于一个类…

【清理mysql数据库服务器二进制日志文件】

清理前后比对 清理前占用 86% &#xff1a; 清理后占用 29% &#xff1a; 排查占用磁盘较大的文件 检测磁盘空间占用 TOP 10 # 检测磁盘空间占用 TOP 10 $ sudo du -S /var/log/ | > sort -rn | # -n选项允许按数字排序。-r选项会先列出最大数字&#xff08;逆序&#x…

Tomcat架构分析

Tomcat的核心组件 Tomcat将请求器和处理器分离&#xff0c;使用多种请求器支持不同的网络协议&#xff0c;而处理器只有一个。从而网络协议和容器解耦。 Tomcat的容器 Host&#xff1a;Tomcat提供多个域名的服务&#xff0c;其将每个域名都视为一个虚拟的主机&#xff0c;在…

git忽略某些文件(夹)更改说明

概述 在项目中,常有需要忽略的文件、文件夹提交到代码仓库中,在此做个笔录。 一、在项目根目录内新建文本文件,并重命名为.gitignore,该文件语法如下 # 以#开始的行,被视为注释. # 忽略掉所有文件名是 a.txt的文件. a.txt # 忽略所有生成的 java文件, *.java # a.j…

java演唱会网上订票购票系统springboot+vue

随着计算机技术的发展以及计算机网络的逐渐普及&#xff0c;互联网成为人们查找信息的重要场所&#xff0c;二十一世纪是信息的时代&#xff0c;所以信息的交换和信息流通显得特别重要。因此&#xff0c;开发合适的基于springboot的演唱会购票系统的设计与实现成为企业必然要走…

【MySQL】内置函数 -- 详解

一、日期函数 日期&#xff1a;年月日时间&#xff1a;时分秒 1、获得年月日 2、获得时分秒 3、获得时间戳 4、在日期的基础上加日期 5、在日期的基础上减去时间 6、计算两个日期之间相差多少天 7、获得当前时间 ⚪练习 &#xff08;1&#xff09;记录生日 &#xff08;2&…

Qt中关于信号与槽函数的思考

信号与槽函数的思考 以pushbutton控件为例&#xff0c;在主界面上放置一个pushbutton控件&#xff0c;点击右键选择关联槽函数&#xff0c;关联一个click函数&#xff0c;如下图所示&#xff1a; 在该函数中&#xff0c;实现了一个点击pushbutton按钮后&#xff0c;弹出一个窗…

德人合科技 | 天锐绿盾终端安全管理系统

德人合科技提到的“天锐绿盾终端安全管理系统”是一款专业的信息安全防泄密软件。这款软件基于核心驱动层&#xff0c;为企业提供信息化防泄密一体化方案。 www.drhchina.com 其主要特点包括&#xff1a; 数据防泄密管理&#xff1a;天锐绿盾终端安全管理系统能够确保数据在创…

淘宝商品数据爬取商品信息采集数据分析API接口详细步骤展示(含测试链接)

01 数据采集 数据采集是数据可视化分析的第一步&#xff0c;也是最基础的一步&#xff0c;数据采集的数量和质量越高&#xff0c;后面分析的准确的也就越高&#xff0c;我们来看一下淘宝网的数据该如何爬取。点此获取淘宝API测试key&密钥 淘宝网站是一个动态加载的网站&a…

pytorch 图像的卷积操作

目录 1.卷积核基本参数说明 2.卷积相关操作说明 3.卷积操作示例 1.卷积核基本参数说明 pytorch进行图像卷积操作之前&#xff0c;需要把图像素格式进行分离&#xff0c;比如一个图像为rgb格式&#xff0c;把R&#xff0c;G,B取出来作为一个ndarray&#xff0c;前文讲过&#…

基于串流技术的p2p共享桌面共享方案

研究远控有一定时间了&#xff0c;但真正落地运用的不多&#xff0c;所以也不太上心&#xff0c;平时也只是自己diy玩玩&#xff0c;远程共享看看电视剧。 最近生成式ai大火&#xff0c;直接带动了gpu应用的相关场景&#xff0c;相关场景&#xff0c;但gpu卡又贵&#xff0c;对…

TP6上传图片到OSS(记录贴)

1&#xff0c;先安装&#xff0c;我使用composer安装 在项目的根目录运行composer require aliyuncs/oss-sdk-php 2,安装成功以后vendor目录下可以看到如图&#xff1a; 3&#xff0c;上传图片代码如下&#xff1a; <?php namespace app\controller;use app\BaseControll…