迭代器iterator失效是在使用STL容器特别需要注意的一个问题。迭代器失效主要发生在迭代过程中对容器进行插入或者删除的改动过程中。这里主要记录一个我遇到的特殊的失效样例。
#include <iostream>
#include <set>
using namespace std;
void print_set(set<int> aa) {
for (auto a : aa) {
printf("%d\n", a);
}
}
int main() {
set<int> aa = {1,2,3,4,5,6,7,8,9};
for (auto v : aa) {
if (v == 7) {
aa.erase(v);
}
printf("%ld\n", aa.size());
}
print_set(aa);
return 0;
}
该段代码使用range-based for 循环(C++11之后开始支持)遍历set aa,期间删除aa中的元素7。该段代码可以在Linux g++和 Visual Studio 的 release模式下编译执行。但在 debug模式抛出异常。
引发了异常: 读取访问权限冲突。
this->_Ptr->_Right 是 0xDDDDDDDD。
猜测:可能的原因是 g++ 或者release在编译时一些优化措施提供了容错机制。
(没有深入调查,先记录下,以后补充)
为了避免其中有坑。建议直接使用iterator指针,并在删除之前增加迭代器。
int main() {
set<int> aa = {1,2,3,4,5,6,7,8,9};
for (auto it = aa.begin(); it != aa.end();) {
if (*it == 7) {
aa.erase(it++);
} else {
it++;
}
printf("%ld\n", aa.size());
}
print_set(aa);
return 0;
}
注意后置自增操作的过程,先取值,再+1。类似的,如果在另外的函数中进行erase操作,可以使用一个临时变量(指针),然后先对迭代器+1,再去erase临时变量(指针)对应的位置。