關(guān)于C++中vector的兩個(gè)小tips分享
前言
本來這篇文章標(biāo)題我想起成《關(guān)于 vector 的兩個(gè)小坑》,后來想想,其實(shí)也不算是坑,還是自己對(duì)原理性的東西理解的沒做那么透徹。工作中遇到的很多問題,后來歸根到底都是基礎(chǔ)不牢靠。
vector 擴(kuò)容
這個(gè)問題很經(jīng)典了,但還是不小心踩到。有一個(gè)需求是要對(duì)目標(biāo)元素進(jìn)行復(fù)制,而目標(biāo)元素集合是保存在 vector 里面,于是簡(jiǎn)單思考下就有如下代碼(大致含義):
void Duplidate(vector<Element>* element_list, Element* element) { element_list.push_back(*element); } void Process() { for (auto& package : package_list) { if (IsNeedDuplicate()) { Duplicate(element_list, package->element); } } }
看起來好像沒什么問題,就是當(dāng)前的 package 對(duì)象是否滿足復(fù)制的要求,需要的話,就對(duì) package 的成員 origin_element 進(jìn)行復(fù)制。跑 UT 也正常,然后在測(cè)試的時(shí)候就 coredump 了???core 文件就是掛在了復(fù)制的時(shí)候。這里我一開始就沒明白,一個(gè)簡(jiǎn)單的復(fù)制為什么會(huì)有 coredump。
檢查了很久 element 復(fù)制的場(chǎng)景,甚至想要專門寫一個(gè)拷貝構(gòu)造函數(shù)。最后才恍然大悟, origin_element 指針指向的就是 element_list 里面的元素, element_list 是整體流程的數(shù)據(jù)源, packge 對(duì)象是封裝的中間處理對(duì)象。之前的開發(fā)人員為了方便,直接在 package 對(duì)象上保存了原始的 element 指針,而這個(gè)指針指向的是一個(gè) vector 里的元素。而我新加的邏輯會(huì)往原始的 vector 里面再添加元素,那么就有可能導(dǎo)致 vector 擴(kuò)容,而 vector 擴(kuò)容會(huì)導(dǎo)致整體的復(fù)制,從而導(dǎo)致原來指向這些元素的指針都失效了,靠后的 package 對(duì)象再去訪問 origin_element 就產(chǎn)生了 coredump。
當(dāng)然,從設(shè)計(jì)上來說,就不應(yīng)該保存指向 vector 元素的指針,但是這里有太多舊代碼牽涉,這里就不做討論。
vector::erase()
起因是我在代碼里面新增了如下代碼(大致):
void EraseElement(const vector<Element>::iterator& element_iter, vector<Element>& element_list) { while (element_iter != element_list.end()) { element_list.erase(element_iter); } }
然后 cr 的同學(xué)提出了一個(gè)疑問是 element_iter 是 const 不可變的,但是在函數(shù)里有擦除了對(duì)應(yīng)的元素,這里會(huì)不會(huì)有問題?雖然 UT 都已經(jīng)跑過了,但是這種寫法的確比較奇怪,于是就借機(jī)學(xué)習(xí)了一下 vector::erase() 的實(shí)現(xiàn)原理跟用法。
erase(iterator) 的實(shí)現(xiàn)原理其實(shí)不會(huì)改變 iterator ,而是把后面的元素一個(gè)個(gè)往前移動(dòng),相當(dāng)于是 iterator 指向的元素本身發(fā)生了變化,所以可以用 const 來修飾這個(gè) iterator 。但是這里用 cosnt & 其實(shí)是沒有錯(cuò)但是無用的修飾,除了容易讓人誤判之外,其實(shí)沒有什么實(shí)際用途。我之前是為了修正 cpplint 才把reference 改成 const reference。
另外 erase 本身的確比較危險(xiǎn),主要還是 erase 的時(shí)候 iterator 本身沒發(fā)生變化,但是指向的元素變了,,在很多時(shí)候 iterator 會(huì)自然地指向下一個(gè)元素,但是由于這是未定義的行為,這里面可能會(huì)有不可預(yù)期的地方,所以最終改成顯示的獲取返回重新賦值( erase() 會(huì)返回下一個(gè)迭代器,但這一點(diǎn)常常被忽略),這樣就能保證安全性了。更安全更推薦的做法應(yīng)該是使用 remove_if() 這里就不展開講了。
void EraseElement(vector<Element>& element_list, vector<Element>::iterator element_iter ) { while (element_iter != element_list.end()) { element_iter = element_list.erase(element_iter); } }
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)我們的支持。
欄 目:C語言
下一篇:C語言實(shí)現(xiàn)520表白代碼 祝你表白成功!
本文標(biāo)題:關(guān)于C++中vector的兩個(gè)小tips分享
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/307.html
您可能感興趣的文章
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)數(shù)怎么表達(dá)
- 04-02c語言沒有round函數(shù) round c語言
- 04-02C語言中怎么打出三角函數(shù) c語言中怎么打出三角函數(shù)的值
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10深入Main函數(shù)中的參數(shù)argc,argv的使用詳解
- 01-10APUE筆記之:進(jìn)程環(huán)境詳解
- 01-10c++中inline的用法分析
- 01-10如何尋找數(shù)組中的第二大數(shù)


閱讀排行
- 1C語言 while語句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 04-02jquery與jsp,用jquery
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10delphi制作wav文件的方法