今天同学华为面试被问到vector有什么问题了,我一拍脑门,vector有什么问题??

原来是迭代器失效问题。先看看vector中:

void test_vector_erase(){
    vector<int> v = {1,2,3,4,5};
    for(auto it = v.begin(); it != v.end(); it++){
        if(*it == 3)
            v.erase(it);
    }
    for(int& x : v) cout<<x<<" ";
}

我本来是想以这个例子举例的,但似乎没有bug??给我整不会了。

但是我又分别测试了删除 1 和删除 5 的情况,答案是删除5的时候会出问题

void test_vector_erase(){
    vector<int> v = {1,2,3,4,5};
    for(auto it = v.begin(); it != v.end(); it++){
        if(*it == 5)       //会出问题, 这里改成*it  > 2也会出问题
            v.erase(it);
    }
    for(int& x : v) cout<<x<<" ";
}

 

其实erase操作之后it还是会留在原位置,比如1,2,3,我们删除了2,变成了1,3,it由原来指向变成了指向3,这个时候做--操作,抵消for循环里面的++操作可以得到正确的结果

void modify_vector_erase(){
    vector<int> v = {1,2,3,4,5};
    for(auto it = v.begin(); it != v.end(); it++){
        if(*it > 2) {
            it = v.erase(it);
            it--;
        }
    }
    for(auto it = v.begin(); it != v.end(); it++){
        cout<<*it<<" ";
    }
}

 

再来看看insert操作是如何让迭代器失效的

int test_vector_insert(){
    vector<int> v(3);
    v.reserve(10);  //注意这句话, 不加这句话输出的结果是99 98 0 0 100 3 98 ,可以看出迭代器并不在我们想要的位置,因为insert当空间不够时会引起扩容,原数组被搬到了另一块更大的空间,迭代器还留在原来的位置
  //加上这句话,也就是vector的预分配函数后,我们insert四个数不会引起扩容,因此会打印出正确结果 3 2 1 0 100 99 98 
v[0] = 100;v[1] = 99, v[2] = 98; auto it = v.begin(); v.insert(it, 1, 0); v.insert(it, 1, 1); v.insert(it, 1, 2); v.insert(it, 1, 3); for(auto it = v.begin(); it != v.end(); it++){ cout<<*it<<" "; } return 0; }

 

再看看关联性容器map, 对于map和list,这种存放位置不连续的容器,当客户端对它进行元素新增操作(insert)或者删除(erase)操作时,操作之前的所有迭代器在操作完成后都依然有效,被删除的那个元素的迭代器是个例外。

 首先直接删除任意一个元素后就不会继续遍历后面的元素了,输出1,3,4,5。被删除的it被删除后还是指向3,而此时it的下一个是mp.end()

void test_map_erase(){
    map<int, char> mp{{1,'a'},{2,'b'},{3,'c'},{4,'d'},{5,'e'}};
    for(auto it = mp.begin(); it != mp.end(); it++){
        if(it -> first > 2){
       mp.erase(it);
       cout<<it->first<<endl;
     }     }
for(auto it = mp.begin(); it != mp.end(); it++) cout<<it->first<<" "; }

 

下面这样可以完整删除3,4,5

void test_map_erase(){
    map<int, char> mp{{1,'a'},{2,'b'},{3,'c'},{4,'d'},{5,'e'}};
    for(auto it = mp.begin(); it != mp.end(); it++){
        if(it -> first > 2) {
            mp.erase(it++);
            it--;
        }
    }
    for(auto it = mp.begin(); it != mp.end(); it++)
        cout<<it->first<<" ";
}

 

总之,我觉得不会有哪个傻蛋会一遍修改容器一边遍历容器吧!

另外erase操作和remove操作也是有区别的

 

//remove的官方解释,是把一个范围内的某个数删除,但是并没有释放容器空间
template <class ForwardIterator, class T>
  ForwardIterator remove (ForwardIterator first, ForwardIterator last, const T& val)
{
  ForwardIterator result = first;
  while (first!=last) {
    if (!(*first == val)) {
      if (result!=first)
        *result = move(*first);
      ++result;
    }
    ++first;
  }
  return result;
}

//vector 中的 erase是删除某个位置并且释放了空间的(ps:map的源码我没看到,不知道有没有释放)
iterator erase(iterator position){
    if(position + 1 != end())
        copy(position + 1, finish, position);
    --finish;    //finish是vector中已使用空间的尾,end_of_storage 是vector可使用空间的尾
    destroy(finish);
    return position;
}

 

 

STL共有这五种类型的迭代器,迭代器其实是一种行为类似智能指针的对象。

C++ 迭代器失效问题_预分配       C++ 迭代器失效问题_迭代器失效_02