2

cpp的新手。当我在迭代器上调用擦除而不增加它时,我的程序立即退出,没有任何迹象或崩溃。在下面的示例中,仅打印了“After if condition”,并且没有打印。没有显示错误。但是,如果我正确调用list.erase(it++); 比它打印所有。 我的问题是当我不正确时发生了什么。为什么我没有看到任何崩溃或退出?我的担心和我问这个问题的原因是关于捕捉这些类型的崩溃,为什么我没有在 catch 块中捕捉到它?有没有办法抓住它?

int main(int argc, char *argv[]) {
    try {
        list<int>::iterator it;
        list<int> list;
        list.push_back(1);
        for (it = list.begin(); it != list.end(); ++it) {
            if ((*it) == 1) {
                list.erase(it);
            }
            cerr << "After if condition " << endl;
        }
        cerr << "After for condition" << endl;
    } catch (...) {
        cerr << "catch exception" << endl;
    }
    cerr << "Finish" << endl;
}
4

4 回答 4

6

来自std::list::erase()

对已擦除元素的引用和迭代器无效。

it被擦除,然后通过++it导致未定义的行为来使用:意味着任何事情都可能发生。

erase()返回最后一个删除元素之后的迭代器,这意味着您可以存储返回值erase()并继续,但循环需要修改以避免erase()在执行后跳过元素。

于 2012-07-25T12:32:21.053 回答
2

使用迭代器时,您不能更改容器。erase立即使迭代器无效。

你可以做的是这样的:

for( list<int>::iterator it = list.begin(); it != list.end();)
  if( (*it) == 1 )        
    it=list.erase(it);
  else
    ++it;

(从STL 列表中删除项目

或者

for( list<int>::iterator it = list.begin(); it != list.end();)
  if( (*it) == 1 )        
    list.erase(it++);
  else
    ++it;

有很多帖子。搜索“擦除列表迭代器”!

编辑:如果您不这样做,即使迭代器无效并随后增加,则行为未定义。这意味着它可能因系统、编译器和编译器的不同而不同。

如果它没有崩溃并给出正确的行为,那么你很幸运。如果它没有崩溃并执行其他操作,那么您很不幸,因为这是一个很难找到的错误。

于 2012-07-25T12:28:31.637 回答
2

it抹掉后就不能使用了。调用 ++it 会导致崩溃。在擦除元素之前存储下一个迭代器。

于 2012-07-25T12:29:30.420 回答
0

诀窍是使用erase(it++).

于 2012-07-25T12:29:20.053 回答