天下网商自助建站系统,建设银行跨行转账网站,福州网站建设的公司,建网站公司浩森宇特【C】 vector 迭代器失效问题 一. 迭代器失效问题分析二. 对于vector可能会导致其迭代器失效的操作有#xff1a;1. 会引起其底层空间改变的操作#xff0c;都有可能是迭代器失效2. 指定位置元素的删除操作--erase3. Linux下#xff0c;g编译器对迭代器失效的检测并不是非常… 【C】 vector 迭代器失效问题 一. 迭代器失效问题分析二. 对于vector可能会导致其迭代器失效的操作有1. 会引起其底层空间改变的操作都有可能是迭代器失效2. 指定位置元素的删除操作--erase3. Linux下g编译器对迭代器失效的检测并不是非常严格处理也没有vs下极端。4. 与vector类似string在 插入或 扩容操作 或 erase之后迭代器也会失效 一. 迭代器失效问题分析 迭代器的主要作用就是让算法能够不用关心底层数据结构其底层实际就是一个指针或者是对指针进行了封装比如vector的迭代器就是原生态指针T。因此迭代器失效实际就是迭代器底层对应指针所指向的空间被销毁了而使用一块已经被释放的空间造成的后果是程序崩溃(即如果继续使用已经失效的迭代器 程序可能会崩溃)。 注意1迭代器失效后代码并不一定会崩溃但是运行结果肯定不对如果it不在begin和end范围内肯定会崩溃的。
注意2vector使用动态分配数组来存储它的元素。当新元素插入时候这个数组需要被重新分配大小为了增加存储空间。 “ 其做法是分配一个新的数组然后将全部元素移到这个数组 ”。 但是原来定义的的迭代器未作处理依旧指向原来的地址这就是导致迭代器失效的原因。 也就是说一旦扩容就会导致迭代器失效。 迭代器失效解决办法在使用前对迭代器重新赋值即可 看图分析: 一旦经过扩容后原来的迭代器指针 it 不可在用因为它还指向原来的旧空间旧空间会被释放旧空间释放后 it 就会变为野指针需要重新更新迭代器即 newit
二. 对于vector可能会导致其迭代器失效的操作有
1. 会引起其底层空间改变的操作都有可能是迭代器失效
比如resize、reserve、insert、assign、push_back等。
#include iostream
using namespace std;
#include vector
int main()
{vectorint v{ 1,2,3,4,5,6 };auto it v.begin();// 将有效元素个数增加到100个多出的位置使用8填充操作期间底层会扩容// v.resize(100, 8);// 插入元素期间可能会引起扩容而导致原空间被释放// v.insert(v.begin(), 0);// // v.push_back(8);// 给vector重新赋值可能会引起底层容量改变v.assign(100, 8);while (it ! v.end()){cout *it ;it;}cout endl;return 0;
}出错原因以上操作都有可能会导致vector扩容也就是说vector底层原理旧空间被释放掉而在打印时it还使用的是释放之间的旧空间在对it迭代器操作时实际操作的是一块已经被释放的空间而引起代码运行时崩溃。 解决方式在以上操作完成之后如果想要继续通过迭代器操作vector中的元素只需给it重新赋值即可。 2. 指定位置元素的删除操作–erase
#include iostream
using namespace std;
#include vector
int main()
{int a[] { 1, 2, 3, 4 };vectorint v(a, a sizeof(a) / sizeof(int));// 使用find查找3(第一个)所在位置的iteratorvectorint::iterator pos find(v.begin(), v.end(), 3);// 删除pos位置的数据导致pos迭代器失效。v.erase(pos);cout *pos endl; // 此处会导致非法访问return 0;
}erase删除pos位置元素后pos位置之后的元素会往前搬移没有导致底层空间的改变理论上讲迭代器不应该会失效但是如果pos刚好是最后一个元素删完之后pos刚好是end的位置而end位置是没有元素的那么pos就失效了。因此删除vector中任意位置上元素时vs编译器就认为该位置迭代器失效了。 3. Linux下g编译器对迭代器失效的检测并不是非常严格处理也没有vs下极端。
#include iostream
using namespace std;
#include vector
// 1. 扩容之后迭代器已经失效了程序虽然可以运行但是运行结果已经不对了
int main()
{vectorint v{ 1,2,3,4,5 };for (size_t i 0; i v.size(); i)cout v[i] ;cout endl;auto it v.begin();cout 扩容之前vector的容量为: v.capacity() endl;// 通过reserve将底层空间设置为100目的是为了让vector的迭代器失效 v.reserve(100);cout 扩容之后vector的容量为: v.capacity() endl;// 经过上述reserve之后it迭代器肯定会失效在vs下程序就直接崩溃了但是linux下不会// 虽然可能运行但是输出的结果是不对的while (it ! v.end()){cout *it ;it;}cout endl;return 0;
}4. 与vector类似string在 插入或 扩容操作 或 erase之后迭代器也会失效