實(shí)踐中遇到使用vector<string>作為局部變量臨時(shí)存儲(chǔ)大量string的情況,發(fā)現(xiàn)運(yùn)行后大量?jī)?nèi)存未被釋放,看了一下相關(guān)的問(wèn)題,一般都建議使用swap來(lái)釋放內(nèi)存,請(qǐng)問(wèn)如果不使用swap是否真的會(huì)存在內(nèi)存泄漏嗎?使用vector<string>存儲(chǔ)大量string的最佳實(shí)踐是什么?
vector<int>
無(wú)需釋放內(nèi)存,但是vector<string>
需要swap釋放
當(dāng)然你會(huì)疑惑 int無(wú)需釋放內(nèi)存,string 也無(wú)需釋放內(nèi)存(會(huì)自動(dòng)析構(gòu)釋放),但是為什么在vector中不一樣?vector<int>
等內(nèi)置類(lèi)型無(wú)需釋放內(nèi)存,自動(dòng)釋放。string類(lèi)型的本質(zhì)是指針,vector<string>
,vector<int*>
等指針類(lèi)型需要手動(dòng)swap釋放。
vector<string>析構(gòu)時(shí),vector和string申請(qǐng)的內(nèi)存都會(huì)被正確釋放,不會(huì)引發(fā)內(nèi)存泄漏。但這些內(nèi)存不一定會(huì)被返還給操作系統(tǒng)。即這一部分內(nèi)存已經(jīng)可用,但依然被進(jìn)程持有,參見(jiàn)。注:std::vector<std::string>().swap(x)
在釋放內(nèi)存上的效果和直接析構(gòu)x沒(méi)有區(qū)別,它依賴(lài)于vector的析構(gòu)來(lái)完成內(nèi)存釋放。這種方案可以安全的清空x,并釋放其持有的內(nèi)存,隨后x依然可用。
如果你在vector被析構(gòu)前觀察到大量?jī)?nèi)存未被釋放,那么可能是vector額外持有內(nèi)存導(dǎo)致的(調(diào)用vector::capacity可檢查vector持有多少內(nèi)存)。vector的內(nèi)存管理機(jī)制會(huì)導(dǎo)致這一現(xiàn)象——vector可能不會(huì)在移除元素后釋放內(nèi)存,vector可能會(huì)申請(qǐng)多一倍的內(nèi)存——這是為了降低添加/移除元素時(shí)的拷貝開(kāi)銷(xiāo)。
在需要釋放vector持有的額外內(nèi)存時(shí),標(biāo)準(zhǔn)做法是調(diào)用 vector::shrink_to_fit ??紤]到shrink_to_fit是實(shí)現(xiàn)相關(guān)的,也可以自己編寫(xiě)一個(gè):
std::vector<std::string> a(100, "abc");
a.erase(a.begin(), a.begin() + 10);
{ // shrink to fit
std::vector<std::string> b;
b.reserve(a.size());
std::move(a.begin(), a.end(), std::back_inserter(b));
a = std::move(b);
}
{ // another workaround
a = std::vector<std::string>(std::make_move_iterator(a.begin()), std::make_move_iterator(a.end()));
}
vector這么做是為了避免頻繁分配內(nèi)存和拷貝。因?yàn)関ector設(shè)計(jì)接近數(shù)組,要求數(shù)據(jù)連續(xù)存放。如果刪除元素就釋放內(nèi)存,以后再添加可能會(huì)連續(xù)可用空間不夠,需要移動(dòng)到其他的內(nèi)存區(qū)域。因?yàn)閟tring內(nèi)部的字符串?dāng)?shù)據(jù)在堆里,并沒(méi)有直接放在vector中,而且從vector中刪除時(shí)會(huì)釋放掉數(shù)據(jù)。所以你不必?fù)?dān)心vector沒(méi)有釋放的空間,一般都是很小的。除非你的vector的數(shù)據(jù)量變化幅度極大,一般是不用收縮vector的。
北大青鳥(niǎo)APTECH成立于1999年。依托北京大學(xué)優(yōu)質(zhì)雄厚的教育資源和背景,秉承“教育改變生活”的發(fā)展理念,致力于培養(yǎng)中國(guó)IT技能型緊缺人才,是大數(shù)據(jù)專(zhuān)業(yè)的國(guó)家
北大青鳥(niǎo)中博軟件學(xué)院創(chuàng)立于2003年,作為華東區(qū)著名互聯(lián)網(wǎng)學(xué)院和江蘇省首批服務(wù)外包人才培訓(xùn)基地,中博成功培育了近30000名軟件工程師走向高薪崗位,合作企業(yè)超4
中公教育集團(tuán)創(chuàng)建于1999年,經(jīng)過(guò)二十年潛心發(fā)展,已由一家北大畢業(yè)生自主創(chuàng)業(yè)的信息技術(shù)與教育服務(wù)機(jī)構(gòu),發(fā)展為教育服務(wù)業(yè)的綜合性企業(yè)集團(tuán),成為集合面授教學(xué)培訓(xùn)、網(wǎng)
達(dá)內(nèi)教育集團(tuán)成立于2002年,是一家由留學(xué)海歸創(chuàng)辦的高端職業(yè)教育培訓(xùn)機(jī)構(gòu),是中國(guó)一站式人才培養(yǎng)平臺(tái)、一站式人才輸送平臺(tái)。2014年4月3日在美國(guó)成功上市,融資1
曾工作于聯(lián)想擔(dān)任系統(tǒng)開(kāi)發(fā)工程師,曾在博彥科技股份有限公司擔(dān)任項(xiàng)目經(jīng)理從事移動(dòng)互聯(lián)網(wǎng)管理及研發(fā)工作,曾創(chuàng)辦藍(lán)懿科技有限責(zé)任公司從事總經(jīng)理職務(wù)負(fù)責(zé)iOS教學(xué)及管理工作。
浪潮集團(tuán)項(xiàng)目經(jīng)理。精通Java與.NET 技術(shù), 熟練的跨平臺(tái)面向?qū)ο箝_(kāi)發(fā)經(jīng)驗(yàn),技術(shù)功底深厚。 授課風(fēng)格 授課風(fēng)格清新自然、條理清晰、主次分明、重點(diǎn)難點(diǎn)突出、引人入勝。
精通HTML5和CSS3;Javascript及主流js庫(kù),具有快速界面開(kāi)發(fā)的能力,對(duì)瀏覽器兼容性、前端性能優(yōu)化等有深入理解。精通網(wǎng)頁(yè)制作和網(wǎng)頁(yè)游戲開(kāi)發(fā)。
具有10 年的Java 企業(yè)應(yīng)用開(kāi)發(fā)經(jīng)驗(yàn)。曾經(jīng)歷任德國(guó)Software AG 技術(shù)顧問(wèn),美國(guó)Dachieve 系統(tǒng)架構(gòu)師,美國(guó)AngelEngineers Inc. 系統(tǒng)架構(gòu)師。