C++ 迭代器失效详解:如何避免 vector 操作中的陷阱

目录

1. 什么是迭代器失效?

2. 哪些操作会导致迭代器失效?

2.1 vector 的插入操作(push_back, insert)

示例:push_back 导致迭代器失效

如何避免?

2.2 vector 的删除操作(erase, pop_back)

示例:erase 导致迭代器失效

如何正确删除?

3. 其他容器的迭代器失效情况

4. 总结


1. 什么是迭代器失效?

在 C++ 中,迭代器(iterator) 是一种类似指针的对象,用于遍历 STL 容器(如 vectorlistmap 等)。
迭代器失效 是指在对容器进行某些操作(如插入、删除)后,原本有效的迭代器变得不可用,继续使用它会导致 未定义行为(Undefined Behavior, UB),如程序崩溃、数据错误等

2. 哪些操作会导致迭代器失效?

不同的容器有不同的迭代器失效规则,本文主要讨论 vector 的迭代器失效问题。

2.1 vector 的插入操作(push_backinsert

当向 vector 插入元素时:

  • 如果 size() == capacity()(容量已满)
    • vector 会重新分配更大的内存,并拷贝原有数据。
    • 所有迭代器失效(包括 begin()end() 等)。
  • 如果 size() < capacity()(容量未满)
    • 插入点之前的迭代器仍然有效
    • 插入点及之后的迭代器失效(因为元素可能被移动)。
示例:push_back 导致迭代器失效
vector<int> v = {1, 2, 3};
auto it = v.begin(); // it 指向 1
v.push_back(4);      // 可能触发重新分配内存
cout << *it;         // ❌ 危险!it 可能失效
如何避免?
  • 提前预留空间reserve()):
vector<int> v;
v.reserve(100);    // 预留 100 个元素的空间
auto it = v.begin();
v.push_back(1);    // 不会重新分配,it 仍然有效
  • 使用索引代替迭代器(如果允许)。

2.2 vector 的删除操作(erasepop_back

当从 vector 删除元素时:

  • 被删除元素的迭代器失效
  • 被删除元素之后的所有迭代器失效(因为后面的元素会向前移动)。
  • 删除点之前的迭代器仍然有效
示例:erase 导致迭代器失效
vector<int> v = {1, 2, 3, 4};
auto it = v.begin() + 2; // it 指向 3
v.erase(v.begin() + 1);  // 删除 2
cout << *it;             // ❌ 危险!it 已经失效(3 已经前移)
如何正确删除?
  • 使用 erase 的返回值(返回下一个有效迭代器):
vector<int> v = {1, 2, 3, 4};
auto it = v.begin();
while (it != v.end()) {if (*it % 2 == 0) {it = v.erase(it); // 删除并更新 it} else {it++;             // 否则正常递增}
}

反向遍历(避免迭代器失效)

for (auto it = v.rbegin(); it != v.rend(); ) {if (*it % 2 == 0) {it = vector<int>::reverse_iterator(v.erase(it.base() - 1));} else {it++;}
}

3. 其他容器的迭代器失效情况

容器插入操作(insert删除操作(erase
vector可能失效(取决于容量)被删除及后面的失效
deque可能失效(首尾安全)被删除及附近的失效
list不会失效仅被删除的失效
map/set不会失效仅被删除的失效

4. 总结

  • vector 插入时
    • 可能失效(如果触发重新分配)。
    • 避免方法:提前 reserve() 或使用索引。
  • vector 删除时
    • 被删除及后面的迭代器失效
    • 正确做法:使用 erase 返回值或反向遍历。
  • 其他容器(如 listmap)通常更安全,但仍需谨慎。

最佳实践

  1. 避免在遍历时直接修改容器,除非明确知道迭代器是否有效。
  2. 尽量使用 range-based for 或算法(如 remove_if,减少手动管理迭代器。
  3. 调试时使用 -D_GLIBCXX_DEBUG(GCC)检测迭代器错误。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

(EtherCAT 转 EtherNet/IP)EtherCAT/Ethernet/IP/Profinet/ModbusTCP协议互转工业串口网关

型号 协议转换通信网关 EtherCAT 转 EtherNet/IP MS-GW12 概述 MS-GW12 是 EtherCAT 和 EtherNet/IP 协议转换网关&#xff0c;为用户提供两种不同通讯协议的 PLC 进行数据交互的解决方案&#xff0c;可以轻松容易将 EtherNet/IP 网络接入 EtherCAT 网络中&#xff0c;方便…

榕壹云酒水定制系统:基于THinKPHP+MySQL+UniApp打造数字化时代的个性化购酒新体验

数字化浪潮下的酒水定制新机遇 在消费升级与个性化需求崛起的背景下&#xff0c;传统酒水行业正面临数字化转型的迫切需求。为此&#xff0c;我们团队基于ThinkPHPMySQLUniApp技术栈&#xff0c;开发了一套榕壹云酒水定制系统&#xff0c;旨在通过数字化手段解决消费者个性化购…

GR00T N1:面向通用类人机器人的开放基础模型

摘要 通用型机器人需要具备多功能的身体和智能的大脑。近年来&#xff0c;类人机器人的发展在构建人类世界中的通用自主性硬件平台方面展现出巨大潜力。一个经过大量多样化数据源训练的机器人基础模型&#xff0c;对于使机器人能够推理新情况、稳健处理现实世界的多变性以及快…

WebRTC实时通话EasyRTC嵌入式音视频通信SDK,构建智慧医疗远程会诊高效方案

一、方案背景 当前医疗领域&#xff0c;医疗资源分布不均问题尤为突出&#xff0c;大城市和发达地区优质医疗资源集中&#xff0c;偏远地区医疗设施陈旧、人才稀缺&#xff0c;患者难以获得高质量的医疗服务&#xff0c;制约医疗事业均衡发展。 EasyRTC技术基于WebRTC等先进技…

深入理解主成分分析(PCA):原理、算法与应用

内容摘要 本文深入剖析主成分分析&#xff08;PCA&#xff09;技术。介绍其通过正交变换简化数据维度的核心原理&#xff0c;详细推导基于最小投影距离和最大投影方差的算法过程&#xff0c;总结算法流程步骤。全面分析PCA的优缺点&#xff0c;并对比其与KPCA的差异。同时阐述…

uniapp-商城-25-顶部模块高度计算

计算高度&#xff1a; 使用computed进行顶部模块的计算。 总高度&#xff1a;bartotalHeight log 介绍--收款码这一条目 也就是上一章节的title的高度计算 bodybarheight。 在该组件中&#xff1a; js部分的代码&#xff1a; 包含了导出的名字&#xff1a; shop-head…

【MCP】第一篇:MCP协议深度解析——大模型时代的“神经连接层“架构揭秘

【MCP】第一篇&#xff1a;MCP协议深度解析——大模型时代的"神经连接层"架构揭秘 一、什么是MCP&#xff1f;二、为什么需要MCP&#xff1f;三、MCP的架构四、MCP与AI交互的原理4.1 ReAct&#xff08;Reasoning Acting&#xff09;模式4.2 Function Calling 模式 五…

李飞飞团队新作WorldScore:“世界生成”能力迎来统一评测,3D/4D/视频模型同台PK

从古老神话中对世界起源的幻想&#xff0c;到如今科学家们在实验室里对虚拟世界的构建&#xff0c;人类探索世界生成奥秘的脚步从未停歇。如今&#xff0c;随着人工智能和计算机图形学的深度融合&#xff0c;我们已站在一个全新的起点&#xff0c;能够以前所未有的精度和效率去…

[react]Next.js之自适应布局和高清屏幕适配解决方案

序言 阅读前首先了解即将要用到的两个包的作用 1.postcss-pxtorem 自动将 CSS 中的 px 单位转换为 rem 单位按照设计稿尺寸直接写 px 值&#xff0c;由插件自动计算 rem 值 2.amfe-flexible 动态设置根元素的 font-size&#xff08;即 1rem 的值&#xff09;根据设备屏幕宽度和…

C# 如何比较两个List是否相等?

简介 在 C# 里&#xff0c;比较两个 List 是否相等&#xff0c;需要考虑多个方面&#xff0c;例如列表中的元素顺序、元素本身是否相等。下面介绍几种常见的比较方法&#xff1a; 基本类型比较&#xff08;元素顺序必须一致&#xff09; var list1 new List<int> { 1…

【技术派后端篇】Redis分布式锁:原理、实践与应用

在当今的高并发系统中&#xff0c;分布式锁是保障数据一致性和系统稳定性的重要手段。今天&#xff0c;我们就来深入探讨一下Redis分布式锁&#xff0c;揭开它神秘的面纱。 1 本地锁与分布式锁的区别 在Java开发的早期阶段&#xff0c;我们接触过synchronized和Lock锁&#x…

奥比中光tof相机开发学习笔记

针对奥比中光 tof相机&#xff0c;官方提供的资料如下ProcessOn Mindmap|思维导图 Orbbec SDK Python Wrapper基于Orbbec SDK进行设计封装&#xff0c;主要实现数据流接收&#xff0c;设备指令控制。下面就其开发适配进行如下总结&#xff1a; &#xff08;1&#xff09;系统配…

如何学习嵌入式

写这个文章是用来学习的,记录一下我的学习过程。希望我能一直坚持下去,我只是一个小白,只是想好好学习,我知道这会很难&#xff0c;但我还是想去做&#xff01; 本文写于&#xff1a;2025.04.16 请各位前辈能否给我提点建议&#xff0c;或者学习路线指导一下 STM32单片机学习总…

2025 年蓝桥杯 Java B 组真题解析分享

今年是我第二次参加蓝桥杯软件类Java B组的比赛&#xff0c;虽然赛前做了不少准备&#xff0c;但真正坐在考场上时&#xff0c;还是有种熟悉又紧张的感觉。蓝桥杯的题目一向以“基础创新”著称&#xff0c;今年也不例外&#xff0c;每道题都考验着我们对算法的理解、代码实现能…

Vue3服务器端渲染深度实践:架构、性能与全栈集成

一、SSR架构设计模式 1.1 架构模式选择矩阵 维度CSRSSR混合渲染首次内容渲染(FCP)慢(依赖JS执行)快(HTML直出)按路由动态选择SEO支持需预渲染原生支持关键页预渲染服务端压力低(静态托管)高(实时渲染)使用缓存中间层TTI(可交互时间)受限于JS体积需等待Hydration渐进式激活适用…

2025年泰迪杯数据挖掘竞赛B题论文首发+问题一二三四代码分享

料 基于穿戴装备的身体活动监测 摘要 随着科技的进步&#xff0c;加速度计&#xff0c;能够实时、准确地捕捉人体的动态变化&#xff0c;成为医学应用中的一个重要工具。本文将基于题目收集数据进行相关研究。 针对题目给出的数据集&#xff0c;我们首先进行数据清洗工作。首…

国内AI搜索平台与ChatGPT横向对比分析

一、核心技术差异 1、‌百度文小言‌ 基于文心大模型4.0升级&#xff0c;主打“新搜索”能力&#xff0c;支持多模态输入&#xff08;语音、图片、视频&#xff09;和富媒体搜索结果‌。 独有的“记忆个性化”功能可结合用户历史行为优化回答&#xff0c;并在医疗、教育等垂直…

安卓环境搭建开发工具下载Gradle下载

1.安装jdk(使用java语言开发安卓app) 核心库 java.lang java.util java.sq; java.io 2.安装开发工具(IDE)android studio https://r3---sn-2x3elnel.gvt1-cn.com/edgedl/android/studio/install/2023.3.1.18/android-studio-2023.3.1.18-windows.exe下载完成后一步一步安装即…

Python 趣味学习 -数据类型脱口秀速记公式 [特殊字符]

&#x1f3a4; Python数据类型脱口秀速记公式 &#x1f40d; 1️⃣ 四大金刚登场 "Set叔(无序洁癖)、Tuple爷(顽固老头)、List姐(百变女王)、Dict哥(万能钥匙)"2️⃣ 特性对比RAP &#x1f3b6; 内存/作用域&#xff1a; 全局变量 → 函数内修改 → 可变(mutable)会…

单片机 | 基于51单片机的倾角测量系统设计

以下是一个基于51单片机的倾角测量系统设计详解,包含原理、公式和完整代码: 一、系统原理 核心器件:MPU6050(集成3轴加速度计+陀螺仪) 主控芯片:STC89C52RC(51单片机) 显示模块:LCD1602液晶 工作原理: 通过MPU6050采集XYZ三轴加速度数据,利用重力加速度分量计算俯仰…