青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Hello world

常用鏈接

統(tǒng)計(jì)

最新評論

小心刪除容器中元素時(shí)的迭代器失效

 

 從一個(gè)簡單的問題開始,刪除數(shù)組中某個(gè)元素后連續(xù)重復(fù)的元素,例如 1,1,2,3,3,1,1,1,4,0 ---> 1, 2,3,1,4,0。

考慮了幾秒,然后就開始動手寫代碼了:

#include <iostream>
#include 
<vector>
using namespace std;

int main(int argc, char* argv[])
{
   
int a[] = {11333241110};
   
int size = sizeof(a)/sizeof(a[0]);

   vector
<int> vec(a, a+size);

   vector
<int>::iterator iter, end;
   
int previous = vec[0];
   
for (iter = vec.begin() + 1, end = vec.end(); iter != end; ++iter)
   
{        
      
if(*iter == previous)
      
{
         vec.erase(iter);
      }

      
else
      
{
         previous 
= *iter;
      }
    
   }


   
for(iter = vec.begin(); iter != vec.end(); ++iter)
   
{
      cout 
<< *iter << endl;
   }

   
   
return 0;
}


可是編譯一下,出來一大堆error,仔細(xì)看一下出錯(cuò)信息,哦,原來自己忘記了,erase容器中元素的時(shí)候,迭代器會失效。。。頓時(shí)一身冷汗,自己平時(shí)迭代容器的時(shí)候,一般都保存了容器的end元素,要是此時(shí)迭代器失效。。。

Container<int>::iterator iter, end;
for (iter = container.begin() + 1, end = container.end(); iter != end; ++iter)


于是找到收藏的Effective STL,翻開條款9,找到了erase容器中元素的原則。以前曾經(jīng)看過,不過年深日久,早就忘得一干二凈了?,F(xiàn)在還是把要點(diǎn)總結(jié)一下,記在blog上,供以后參考。

1. 對于關(guān)聯(lián)容器(如map, set, multimap,multiset),刪除當(dāng)前的iterator,僅僅會使當(dāng)前的iterator失效,只要在erase時(shí),遞增當(dāng)前iterator即可。這是因?yàn)閙ap之類的容器,使用了紅黑樹來實(shí)現(xiàn),插入、刪除一個(gè)結(jié)點(diǎn)不會對其他結(jié)點(diǎn)造成影響。

for (iter = cont.begin(); it != cont.end();)
{
   (
*iter)->doSomething();
   
if (shouldDelete(*iter))
      cont.erase(iter
++);
   
else
      
++iter;
}

因?yàn)閕ter傳給erase方法的是一個(gè)副本,iter++會指向下一個(gè)元素。

2. 對于序列式容器(如vector,deque),刪除當(dāng)前的iterator會使后面所有元素的iterator都失效。這是因?yàn)関etor,deque使用了連續(xù)分配的內(nèi)存,刪除一個(gè)元素導(dǎo)致后面所有的元素會向前移動一個(gè)位置。還好erase方法可以返回下一個(gè)有效的iterator。

for (iter = cont.begin(); iter != cont.end();)
{
   (
*it)->doSomething();
   
if (shouldDelete(*iter))
      iter 
= cont.erase(iter); 
   
else
      
++iter;
}

3. 對于list來說,它使用了不連續(xù)分配的內(nèi)存,并且它的erase方法也會返回下一個(gè)有效的iterator,因此上面兩種方法都可以使用。


最后給出開始那個(gè)問題的一個(gè)正確的實(shí)現(xiàn):

#include <iostream>
#include 
<vector>
using namespace std;

int main(int argc, char* argv[])
{
   
int a[] = {11333241110};
   
int size = sizeof(a)/sizeof(a[0]);

   vector
<int> vec(a, a+size);

   vector
<int>::iterator iter = vec.begin();
   
int previous = *iter;
   
++iter;
   
for (; iter != vec.end();)
   
{        
      
if(*iter == previous)
      
{
         iter 
= vec.erase(iter);
      }

      
else
      
{
         previous 
= *iter;
         
++iter;
      }
    
   }


   
for(iter = vec.begin(); iter != vec.end(); ++iter)
   
{
      cout 
<< *iter << endl;
   }

   
   
return 0;
}

PS. 不過實(shí)際上這個(gè)問題,用vector來實(shí)現(xiàn)不是很適合,因?yàn)槊看蝿h除一個(gè)元素,都會引起vector的一個(gè)resize操作。resize的時(shí)間復(fù)雜度是O(n),整個(gè)的resize操作要花費(fèi)O(n^2)。最好是選擇list最為容器,list最適合那些需要在容器中間做插入、刪除的例子。

PS2. 另外一個(gè)可能的方法是remove_if + erase慣用法。remove_if的時(shí)間復(fù)雜度是O(n). erase的時(shí)間復(fù)雜度也是O(n),所以整個(gè)操作可以在O(n)時(shí)間里完成。(當(dāng)然,對于這個(gè)例子,ShouldDeletePred比較難寫)

iterator new_end = remove_if(array.begin(), array.end(), ShouldDeletePred());
array.erase(new_end, array.end());

posted on 2009-10-14 19:47 Johnson 閱讀(8058) 評論(3)  編輯 收藏 引用 所屬分類: C++

評論

# re: 小心刪除容器中元素時(shí)的迭代器失效 2009-10-30 16:13 金慶

是不是就是unique()啊?  回復(fù)  更多評論   

# re: 小心刪除容器中元素時(shí)的迭代器失效 2009-11-02 15:25 Johnson

unique會刪除所有符合條件的元素,不是連續(xù)重復(fù)的元素  回復(fù)  更多評論   

# re: 小心刪除容器中元素時(shí)的迭代器失效[未登錄] 2015-05-25 17:39 a

@Johnson
unique好像是的額?  回復(fù)  更多評論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲国产精品va在线观看黑人| 欧美日韩一区二区高清| 欧美日韩在线播放| 日韩午夜剧场| 久久成人综合网| 亚洲一级黄色| 午夜在线观看欧美| 伊人激情综合| 欧美日韩在线播放一区二区| 欧美在线黄色| 亚洲国产经典视频| 这里只有精品在线播放| 国产女精品视频网站免费| 欧美国产欧美亚洲国产日韩mv天天看完整 | 欧美人与性禽动交情品 | 欧美精品一区在线| 欧美日韩国产综合新一区| 欧美日韩美女一区二区| 国产欧美日韩亚洲| 亚洲第一综合天堂另类专| 中国av一区| 久久永久免费| 亚洲精品永久免费| 久久九九99| 国产精品久久久久久影视| 一色屋精品视频免费看| 亚洲一区二区三| 蘑菇福利视频一区播放| 一区二区动漫| 毛片基地黄久久久久久天堂| 国产精品看片资源| 99riav久久精品riav| 久久精品综合一区| 中文一区字幕| 久久精品一级爱片| 亚洲婷婷国产精品电影人久久| 久久av在线看| 亚洲精品日韩精品| 老司机午夜精品视频| 国产精品综合网站| 亚洲一区二区三区午夜| 亚洲国产aⅴ天堂久久| 欧美一区二区三区视频| 欧美午夜不卡| 在线亚洲一区| 亚洲欧洲在线观看| 久久综合中文字幕| 极品尤物av久久免费看| 久久久91精品国产一区二区精品| 99日韩精品| 欧美日本一区二区三区| 亚洲日本中文字幕区| 欧美成人自拍视频| 久久久噜噜噜久久中文字免| 国产亚洲a∨片在线观看| 欧美在现视频| 性高湖久久久久久久久| 国产日韩亚洲欧美精品| 久久精品日韩一区二区三区| 亚洲欧美日韩国产综合在线| 国产精品一区一区| 久久国产精品99精品国产| 亚洲综合三区| 国内精品久久久久久| 看欧美日韩国产| 老色鬼精品视频在线观看播放| 在线激情影院一区| 亚洲国产成人精品女人久久久 | 国产精品国产福利国产秒拍| 亚洲视频www| 亚洲午夜未删减在线观看| 国产精品美女诱惑| 久久久久久一区| 老司机成人网| 亚洲小视频在线| 亚洲欧洲av一区二区| 好吊日精品视频| 亚洲福利小视频| 欧美日韩免费高清| 欧美在线网址| 美女网站久久| 亚洲免费中文| 久久久亚洲精品一区二区三区 | 国产精品久久亚洲7777| 欧美影视一区| 一区二区三区国产精品| 久久亚洲一区二区三区四区| 国产精品一区久久久| 久久久精品性| 欧美福利一区二区三区| 亚洲桃色在线一区| 亚洲欧美国产高清| 亚洲品质自拍| 午夜精品久久久久久久白皮肤| 狠狠色丁香久久综合频道| 亚洲国产一成人久久精品| 国产精品高清免费在线观看| 久久久久国产精品麻豆ai换脸| 欧美成人亚洲| 久久精品亚洲一区二区三区浴池| 噜噜噜在线观看免费视频日韩| 亚洲天堂黄色| 久久久久久高潮国产精品视| 中日韩美女免费视频网址在线观看| 亚洲欧美国产va在线影院| 最新国产拍偷乱拍精品| 午夜视频久久久| 亚洲伦理网站| 欧美专区日韩视频| 亚洲午夜精品久久久久久浪潮| 久久精品二区亚洲w码| 亚洲视频在线观看视频| 久久在线视频| 久久久国产精品亚洲一区| 欧美区高清在线| 欧美黄色小视频| 经典三级久久| 欧美一区二区三区精品| 亚洲欧美经典视频| 欧美日韩亚洲系列| 亚洲日本成人| 亚洲精品久久久久| 麻豆精品在线播放| 久久尤物电影视频在线观看| 国产欧亚日韩视频| 亚洲小视频在线观看| 亚洲视频精品在线| 欧美日韩国产在线一区| 亚洲人精品午夜在线观看| 亚洲精品国精品久久99热一| 久久综合九色综合网站| 欧美成人国产va精品日本一级| 国语自产在线不卡| 久久色在线播放| 免播放器亚洲| 亚洲国产精品成人一区二区| 久久一区二区三区超碰国产精品| 久久精品在线免费观看| 黄色亚洲在线| 久久午夜视频| 亚洲成色777777女色窝| 亚洲大胆在线| 欧美精品一区视频| 99精品免费视频| 亚洲综合大片69999| 国产精品久久久99| 亚洲欧美国产va在线影院| 久久国产精品久久久久久久久久| 国产日产欧美精品| 久久综合成人精品亚洲另类欧美| 欧美国产三级| 欧美sm重口味系列视频在线观看| 激情欧美日韩| 榴莲视频成人在线观看| 亚洲人成在线播放| 亚洲欧美在线另类| 国产一区二区三区自拍| 卡通动漫国产精品| 夜久久久久久| 久久久久五月天| 亚洲精品一区二区三区av| 国产精品久久久99| 久久影院午夜论| 日韩亚洲欧美一区二区三区| 午夜国产精品视频| 在线观看91精品国产麻豆| 欧美激情精品久久久久久蜜臀| 一区二区三区鲁丝不卡| 久久亚洲春色中文字幕久久久| 亚洲国产精品尤物yw在线观看| 欧美久久久久久久| 亚洲欧美亚洲| 亚洲第一区中文99精品| 亚洲午夜羞羞片| 狠狠久久婷婷| 欧美日韩天堂| 久久婷婷麻豆| 亚洲一二区在线| 欧美国产在线观看| 欧美一区二区三区在线看 | 国产一区二区三区四区老人| 免费欧美在线视频| 午夜在线成人av| 亚洲日韩欧美视频| 久久性色av| 亚洲你懂的在线视频| 亚洲日韩欧美视频| 狠狠干狠狠久久| 国产精品爽爽ⅴa在线观看| 欧美成人性生活| 久久五月激情| 久久精品日韩欧美| 香蕉国产精品偷在线观看不卡| 亚洲精品免费网站| 牛牛国产精品| 久久这里有精品视频| 性欧美videos另类喷潮| 亚洲视频电影在线| 99国内精品久久| 亚洲黄色av|