当前位置: 首页 > news >正文

c++STL 迭代器失效的三种情况总结

c++STL 迭代器失效的三种情况总结

参考

1.序列式容器(如 vector, deque)中迭代器失效

因为 vetor、deque 使用了连续分配的内存,erase操作删除一个元素导致后面所有的元素都会向前移动一个位置,这些元素的地址发生了变化,所以当前位置到容器末尾元素的所有迭代器全部失效。

#include<iostream>
#include<vector>

using namespace std;

int main()
{
	vector<int> q;

	for (int i = 0;i<10;i++)
	{
		q.push_back(i);
	}
	
	// 在这里想把大于5的元素都删除
	vector<int>::iterator   it = q.begin();
	//注意这里使用while条件循环判断,不使用for循环
	while (it != q.end())
	{
		//1.正确版本
		if (*it > 5)
		{
			it = q.erase(it); //返回删除后的第一个元素位置(地址),(删除3的指针,返回4的指针)
		}else{
			it++;
		}

		//2.错误版本
		//if (*it > 2)
		//{
		//	q.erase(it); //迭代器失效
		//}
	}
	
	// 打印结果
	for (vector<int>::iterator  it = q.begin(); it != q.end(); it++) {
		cout << *it << " ";
	}
	cout << endl;

	return 0;
}

2.链表式容器(如 list)中迭代器失效

链表式容器(如 list),删除当前的 iterator,仅仅会使当前的 iterator 失效,
这是因为 list 之类的容器,使用了链表来实现,插入、删除一个结点不会对
其他结点造成影响。只要在 erase 时,递增当前 iterator 即可,并且 erase 方法可以返回下一个有效的 iterator。

#include <iostream>
#include <list>

using namespace std;


int main()
{
	list<int> l1;
	for(int i = 0;i<10;i++)
	{
		l1.push_back(i);
	}

	list<int>::iterator iter;
	iter = l1.begin();
	while(iter != l1.end())
	{
		//1.正确版本
		//if (*iter  > 3)
		//{
		//	l1.erase(iter++);
		//}else
		//{
		//	iter++;
		//}
		
		//2.正确版本
		if (*iter  > 3)
		{
			iter = l1.erase(iter);
		}else
		{
			iter++;
		}

		//3.错误版本
		//if (*iter  > 3)
		//{
		//	l1.erase(iter++);
		//}
	}
	for(iter = l1.begin() ; iter != l1.end() ; iter++)
	{
		cout<<*iter<<" ";
	}
	cout<<endl;

	return 0;
}

3.关联容器(如 map, set,multimap,multiset)中迭代器失效

对于关联容器(如 map, set,multimap,multiset),删除当前的 iterator,仅仅会使当前的 iterator 失效,
只要在 erase 时,递增当前 iterator 即可。这是因为 map 之类的容器,使用了红黑树来实现,插入或者删除
一个结点不会对其他结点造成影响。erase 迭代器只是被删元素的迭代器失效,但是返回值
为 void,所以要采用erase(iter++)的方式删除迭代器,即删除之前先递增。

#include <iostream>
#include <sstream>//->stringstream
#include <map>

using namespace std;

int main()
{
	map<int, string> dataMap;
	for (int i = 0; i < 10; i++)
	{
		string strValue = "Hello, World";

		stringstream ss;
		ss<<i;
		string tmpStrCount;
		ss>>tmpStrCount;
		strValue += tmpStrCount;
		dataMap.insert(make_pair(i, strValue));
	}

	cout<<"Map元素内容为:"<<endl;
	map<int, string>::iterator iter;
	for (iter = dataMap.begin(); iter != dataMap.end(); iter++)
	{
		int nKey = iter->first;
		string strValue = iter->second;
		cout<<strValue<<endl;
	}

	cout<<"内容开始删除:"<<endl;
	//删除操作引发迭代器失效
	iter = dataMap.begin();
	while(iter != dataMap.end())
	{
		int nKey = iter->first;
		string strValue = iter->second;

		if (nKey % 2 == 0)
		{
			//1.正确版本
			//map<int, string>::iterator tmpIter = iter;
			//iter++;
			//dataMap.erase(tmpIter);

			//2.正确版本
			dataMap.erase(iter++);//这样也行

			//3.错误版本
			//dataMap.erase(iter);
		}else{
			iter++;
		}
	}

	//输出信息
	for (iter = dataMap.begin(); iter != dataMap.end(); iter++)
	{
		int nKey = iter->first;
		string strValue = iter->second;
		cout<<strValue<<endl;
	}

	return 0;
}

相关文章:

  • cordova 打包android app
  • 【稀里糊涂学Spring MVC】Filter
  • HK-WEKA如何为勒索软件保护和业务连续性提供支持?
  • springboot+mybaties-plus自动建表
  • 企业IP地址跟踪
  • C++ 小游戏 视频及资料集(3)
  • 十、ThreadPoolExecutor 手撕核心源码
  • Refind多引导系统界面
  • 变身小小科学家 南瓜科学让孩子爱上实验
  • 分布式之ZooKeeper概述
  • app毕业设计开题报告基于Uniapp实现的移动端的医生寻访平台计算机毕业设计
  • 分布式链路追踪技术 Sleuth +Zipkin
  • wsl kali-linux 安装记录
  • SpringBoot整合minio分布式文件实操
  • 电商API按关键字搜索商品
  • 【跃迁之路】【463天】刻意练习系列222(2018.05.14)
  • Android路由框架AnnoRouter:使用Java接口来定义路由跳转
  • AngularJS指令开发(1)——参数详解
  • CSS3 聊天气泡框以及 inherit、currentColor 关键字
  • CSS盒模型深入
  • ES6之路之模块详解
  • ES学习笔记(12)--Symbol
  • Java知识点总结(JavaIO-打印流)
  • Laravel 实践之路: 数据库迁移与数据填充
  • nfs客户端进程变D,延伸linux的lock
  • Otto开发初探——微服务依赖管理新利器
  • Redis 懒删除(lazy free)简史
  • Spring Boot快速入门(一):Hello Spring Boot
  • SQLServer插入数据
  • UEditor初始化失败(实例已存在,但视图未渲染出来,单页化)
  • vue总结
  • 阿里云应用高可用服务公测发布
  • 安装python包到指定虚拟环境
  • 大整数乘法-表格法
  • 技术发展面试
  • 理解在java “”i=i++;”所发生的事情
  • 使用 Xcode 的 Target 区分开发和生产环境
  • 手机端车牌号码键盘的vue组件
  •  一套莫尔斯电报听写、翻译系统
  • [地铁译]使用SSD缓存应用数据——Moneta项目: 低成本优化的下一代EVCache ...
  • zabbix3.2监控linux磁盘IO
  • (30)数组元素和与数字和的绝对差
  • (42)STM32——LCD显示屏实验笔记
  • (day6) 319. 灯泡开关
  • (delphi11最新学习资料) Object Pascal 学习笔记---第8章第2节(共同的基类)
  • (入门自用)--C++--抽象类--多态原理--虚表--1020
  • ./和../以及/和~之间的区别
  • .htaccess配置重写url引擎
  • .net 4.0 A potentially dangerous Request.Form value was detected from the client 的解决方案
  • .net 受管制代码
  • .NET/C# 中你可以在代码中写多个 Main 函数,然后按需要随时切换
  • .net实现客户区延伸至至非客户区
  • @media screen 针对不同移动设备
  • @vue/cli 3.x+引入jQuery
  • [ABC294Ex] K-Coloring