迭代器失效问题
Posted TZC⑥
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了迭代器失效问题相关的知识,希望对你有一定的参考价值。
迭代器指针失效的两种情况:
案例一(insert):
情况一:意义变了
在Linux环境下:
示例:
在vs2019环境下:
可以看到pos是在插入30之前给的值,插入后,可以认为pos就失效了,
在vs2019环境下,插入后pos的意义变了,pos指向的是30,不再是3,导致erase(pos)没有达到删除3的目的,程序崩溃
而在Linux环境下,“正常运行”,因为不同库的所执行的标准不同,检查机制也不一样,vs检查比Linux严格
情况二:野指针
只讲程序又插入了一个数据6,程序也崩了,但这原理跟情况一不同
插入6后,程序进行了增容:
重新开辟了一个capacity=9(vs1.5倍增容)的空间,而pos还指向的是原来空间的地址,该空间已经释放,就出现野指针的问题
所以pos在Insert后就失效了,尽量不要使用它
解决方法:
其实很简单,根本问题就是pos指向了错误的空间,只需再次查找pos的空间即可
案例二(erase):
插入12345删除其中的偶数,下面代码有什么问题?
#include<iostream>
#include <vector>
using namespace std;
int main()
{
vector<int> v;
v.push_back(1);
v.push_back(2);
v.push_back(3);
v.push_back(4);
v.push_back(5);
vector<int>::iterator it = v.begin();
while (it != v.end())
{
if (*it % 2 == 0)
{
v.erase(it);
}
++it;
}
for (auto e : v)
{
cout << e << " ";
}
cout << endl;
return 0;
}
在Linux环境下:
看似运行结果正常
但我们在后面再加入一个偶数6呢?
出现段错误
那这是为什么呢?
可以看到在一次it=2的循环后,直接跳过了it=3,到了it=4
所以在it=6时
出现it与end错开的情况,之后it越界一直往后走,所以段错误
在VS2019下无论末尾是基数还是偶数都会报错,因为VS下erase后vs对++做了检查
解决方法:
只需改动一下while循环
erase会返回删除后的下一个位置
所以该程序程序存在潜在问题,侧面说明vs的检测比linux更严格
insert和erase都会使迭代器失效:
1.insert或erase后迭代器意义变了
2.insert或erase后it成为野指针 (增/缩容)
所以迭代器失效的根本问题就是指向了错误的空间,解决就将它指向正确的空间即可
以上是关于迭代器失效问题的主要内容,如果未能解决你的问题,请参考以下文章