當(dāng)前位置:首頁 > 芯聞號 > 充電吧
[導(dǎo)讀]C++智能指針及其簡單實(shí)現(xiàn)  本文將簡要介紹智能指針shared_ptr和unique_ptr,并簡單實(shí)現(xiàn)基于引用計(jì)數(shù)的智能指針。使用智能指針的緣由1. 考慮下邊的簡單代碼:int?main() {

C++智能指針及其簡單實(shí)現(xiàn)

  本文將簡要介紹智能指針shared_ptr和unique_ptr,并簡單實(shí)現(xiàn)基于引用計(jì)數(shù)的智能指針。

使用智能指針的緣由

1. 考慮下邊的簡單代碼:


int?main()
{
?????int?*ptr?=?new?int(0);
?????return?0;
}

?  就如上邊程序,我們有可能一不小心就忘了釋放掉已不再使用的內(nèi)存,從而導(dǎo)致資源泄漏(resoure leak,在這里也就是內(nèi)存泄漏)。

2. 考慮另一簡單代碼:


int?main()
{
?????int?*ptr?=?new?int(0);
?????delete?ptr;
?????return?0;
}

  我們可能會(huì)心想,這下程序應(yīng)該沒問題了?可實(shí)際上程序還是有問題。上邊程序雖然最后釋放了申請的內(nèi)存,但ptr會(huì)變成空懸指針(dangling pointer,也就是野指針)??諔抑羔槻煌诳罩羔槪╪ullptr),它會(huì)指向“垃圾”內(nèi)存,給程序帶去諸多隱患(如我們無法用if語句來判斷野指針)。

  上述程序在我們釋放完內(nèi)存后要將ptr置為空,即:


ptr?=?nullptr;

  除了上邊考慮到的兩個(gè)問題,上邊程序還存在另一問題:如果內(nèi)存申請不成功,new會(huì)拋出異常,而我們卻什么都沒有做!所以對這程序我們還得繼續(xù)改進(jìn)(也可用try...catch...):


#includeusing?namespace?std;

int?main()
{
????int?*ptr?=?new(nothrow)?int(0);
????if(!ptr)
????{
????????cout?<<?"new?fails."
????????return?0;
????}
????delete?ptr;
????ptr?=?nullptr;
????return?0;
}

3. 考慮最后一簡單代碼:


#includeusing?namespace?std;

int?main()
{
????int?*ptr?=?new(nothrow)?int(0);
????if(!ptr)
????{
????????cout?<<?"new?fails."
????????return?0;
????}
????//?假定hasException函數(shù)原型是?bool?hasException()
????if?(hasException())
????????throw?exception();
????
????delete?ptr;
????ptr?=?nullptr;
????return?0;
}

  當(dāng)我們的程序運(yùn)行到“if(hasException())”處且“hasException()”為真,那程序?qū)?huì)拋出一個(gè)異常,最終導(dǎo)致程序終止,而已申請的內(nèi)存并沒有釋放掉。

  當(dāng)然,我們可以在“hasException()”為真時(shí)釋放內(nèi)存:


//?假定hasException函數(shù)原型是?bool?hasException()
if?(hasException())
{
????????delete?ptr;
????????ptr?=?nullptr;
????????throw?exception();
}

  但,我們并不總會(huì)想到這么做。而且,這樣子做也顯得麻煩,不夠人性化?! ?/p>

  如果,我們使用智能指針,上邊的問題我們都不用再考慮,因?yàn)樗家呀?jīng)幫我們考慮到了。

  因此,我們使用智能指針的原因至少有以下三點(diǎn):

  1)智能指針能夠幫助我們處理資源泄露問題;

  2)它也能夠幫我們處理空懸指針的問題;

  3)它還能夠幫我們處理比較隱晦的由異常造成的資源泄露。

智能指針

  自C++11起,C++標(biāo)準(zhǔn)提供兩大類型的智能指針:

  1. Class shared_ptr實(shí)現(xiàn)共享式擁有(shared ownership)概念。多個(gè)智能指針可以指向相同對象,該對象和其相關(guān)資源會(huì)在“最后一個(gè)引用(reference)被銷毀”時(shí)候釋放。為了在結(jié)構(gòu)復(fù)雜的情境中執(zhí)行上述工作,標(biāo)準(zhǔn)庫提供了weak_ptr、bad_weak_ptr和enable_shared_from_this等輔助類。

  2. Class unique_ptr實(shí)現(xiàn)獨(dú)占式擁有(exclusive ownership)或嚴(yán)格擁有(strict ownership)概念,保證同一時(shí)間內(nèi)只有一個(gè)智能指針可以指向該對象。它對于避免資源泄露(resourece leak)——例如“以new創(chuàng)建對象后因?yàn)榘l(fā)生異常而忘記調(diào)用delete”——特別有用。

  注:C++98中的Class auto_ptr在C++11中已不再建議使用。

shared_ptr

  幾乎每一個(gè)有分量的程序都需要“在相同時(shí)間的多處地點(diǎn)處理或使用對象”的能力。為此,我們必須在程序的多個(gè)地點(diǎn)指向(refer to)同一對象。雖然C++語言提供引用(reference)和指針(pointer),還是不夠,因?yàn)槲覀兺仨毚_保當(dāng)“指向?qū)ο蟆钡淖钅┮粋€(gè)引用被刪除時(shí)該對象本身也被刪除,畢竟對象被刪除時(shí)析構(gòu)函數(shù)可以要求某些操作,例如釋放內(nèi)存或歸還資源等等。

  所以我們需要“當(dāng)對象再也不被使用時(shí)就被清理”的語義。Class shared_ptr提供了這樣的共享式擁有語義。也就是說,多個(gè)shared_ptr可以共享(或說擁有)同一對象。對象的最末一個(gè)擁有者有責(zé)任銷毀對象,并清理與該對象相關(guān)的所有資源。

  shared_ptr的目標(biāo)就是,在其所指向的對象不再被使用之后(而非之前),自動(dòng)釋放與對象相關(guān)的資源。

  下邊程序摘自《C++標(biāo)準(zhǔn)庫(第二版)》5.2.1節(jié):


#include#include#include#includeusing?namespace?std;

int?main(void)
{
????//?two?shared?pointers?representing?two?persons?by?their?name
????shared_ptrpNico(new?string("nico"));
????shared_ptrpJutta(new?string("jutta"),
????????????//?deleter?(a?lambda?function)?
????????????[](string?*p)
????????????{?
????????????????cout?<<?"delete?"?<<?*p?<<?endl;
????????????????delete?p;
????????????}
????????);

????//?capitalize?person?names
????(*pNico)[0]?=?'N';
????pJutta->replace(0,?1,?"J");

????//?put?them?multiple?times?in?a?container
????vector<shared_ptr>?whoMadeCoffee;
????whoMadeCoffee.push_back(pJutta);
????whoMadeCoffee.push_back(pJutta);
????whoMadeCoffee.push_back(pNico);
????whoMadeCoffee.push_back(pJutta);
????whoMadeCoffee.push_back(pNico);

????//?print?all?elements
????for?(auto?ptr?:?whoMadeCoffee)
????????cout?<<?*ptr?<<?"?";
????cout?<<?endl;

????//?overwrite?a?name?again
????*pNico?=?"Nicolai";

????//?print?all?elements
????for?(auto?ptr?:?whoMadeCoffee)
????????cout?<<?*ptr?<<?"?";
????cout?<<?endl;

????//?print?some?internal?data
????cout?<<?"use_count:?"?<<?whoMadeCoffee[0].use_count()?<<?endl;

????return?0;
}

  程序運(yùn)行結(jié)果如下:

  

  關(guān)于程序邏輯可見下圖:

  

  關(guān)于程序的幾點(diǎn)說明:

  1)對智能指針pNico的拷貝是淺拷貝,所以當(dāng)我們改變對象“Nico”的值為“Nicolai”時(shí),指向它的指針都會(huì)指向新值。

  2)指向?qū)ο蟆癑utta”的有四個(gè)指針:pJutta和pJutta的三份被安插到容器內(nèi)的拷貝,所以上述程序輸出的use_count為4。

  4)shared_ptr本身提供默認(rèn)內(nèi)存釋放器(default deleter),調(diào)用的是delete,不過只對“由new建立起來的單一對象”起作用。當(dāng)然我們也可以自己定義內(nèi)存釋放器,就如上述程序。不過值得注意的是,默認(rèn)內(nèi)存釋放器并不能釋放數(shù)組內(nèi)存空間,而是要我們自己提供內(nèi)存釋放器,如:


shared_ptrpJutta2(new?int[10],
????????//?deleter?(a?lambda?function)?
????????[](int?*p)
????????{?
????????????delete[]?p;
????????}
????);

?  或者使用為unique_ptr而提供的輔助函數(shù)作為內(nèi)存釋放器,其內(nèi)調(diào)用delete[]:


shared_ptrp(new?int[10],?default_delete());

unique_ptr

  unique_ptr是C++標(biāo)準(zhǔn)庫自C++11起開始提供的類型。它是一種在異常發(fā)生時(shí)可幫助避免資源泄露的智能指針。一般而言,這個(gè)智能指針實(shí)現(xiàn)了獨(dú)占式擁有概念,意味著它可確保一個(gè)對象和其相應(yīng)資源同一時(shí)間只被一個(gè)指針擁有。一旦擁有者被銷毀或變成空,或開始擁有另一個(gè)對象,先前擁有的那個(gè)對象就會(huì)被銷毀,其任何相應(yīng)資源也會(huì)被釋放。

  現(xiàn)在,本文最開頭的程序就可以寫成這樣啦:


#includeusing?namespace?std;

int?main()
{
????unique_ptrptr(new?int(0));
????return?0;
}

智能指針簡單實(shí)現(xiàn)

  基于引用計(jì)數(shù)的智能指針可以簡單實(shí)現(xiàn)如下(詳細(xì)解釋見程序中注釋):


#includeusing?namespace?std;

templateclass?SmartPtr
{
public:
????SmartPtr(T?*p);
????~SmartPtr();
????SmartPtr(const?SmartPtr&orig);????????????????//?淺拷貝
????SmartPtr&?operator=(const?SmartPtr&rhs);????//?淺拷貝
private:
????T?*ptr;
????//?將use_count聲明成指針是為了方便對其的遞增或遞減操作
????int?*use_count;
};

templateSmartPtr::SmartPtr(T?*p)?:?ptr(p)
{
????try
????{
????????use_count?=?new?int(1);
????}
????catch?(...)
????{
????????delete?ptr;
????????ptr?=?nullptr;
????????use_count?=?nullptr;
????????cout?<<?"Allocate?memory?for?use_count?fails."?<<?endl;
????????exit(1);
????}

????cout?<<?"Constructor?is?called!"?<<?endl;
}

templateSmartPtr::~SmartPtr()
{
????//?只在最后一個(gè)對象引用ptr時(shí)才釋放內(nèi)存
????if?(--(*use_count)?==?0)
????{
????????delete?ptr;
????????delete?use_count;
????????ptr?=?nullptr;
????????use_count?=?nullptr;
????????cout?<<?"Destructor?is?called!"?<<?endl;
????}
}

templateSmartPtr::SmartPtr(const?SmartPtr&orig)
{
????ptr?=?orig.ptr;
????use_count?=?orig.use_count;
????++(*use_count);
????cout?<<?"Copy?constructor?is?called!"?<<?endl;
}

//?重載等號函數(shù)不同于復(fù)制構(gòu)造函數(shù),即等號左邊的對象可能已經(jīng)指向某塊內(nèi)存。
//?這樣,我們就得先判斷左邊對象指向的內(nèi)存已經(jīng)被引用的次數(shù)。如果次數(shù)為1,
//?表明我們可以釋放這塊內(nèi)存;反之則不釋放,由其他對象來釋放。
templateSmartPtr&?SmartPtr::operator=(const?SmartPtr&rhs)
{
????//?《C++?primer》:“這個(gè)賦值操作符在減少左操作數(shù)的使用計(jì)數(shù)之前使rhs的使用計(jì)數(shù)加1,
????//?從而防止自身賦值”而導(dǎo)致的提早釋放內(nèi)存
????++(*rhs.use_count);

????//?將左操作數(shù)對象的使用計(jì)數(shù)減1,若該對象的使用計(jì)數(shù)減至0,則刪除該對象
????if?(--(*use_count)?==?0)
????{
????????delete?ptr;
????????delete?use_count;
????????cout?<<?"Left?side?object?is?deleted!"?<<?endl;
????}

????ptr?=?rhs.ptr;
????use_count?=?rhs.use_count;
????
????cout?<<?"Assignment?operator?overloaded?is?called!"?<<?endl;
????return?*this;
}

  測試程序如下:


#include#include?"smartptr.h"
using?namespace?std;

int?main()
{
????//?Test?Constructor?and?Assignment?Operator?Overloaded
????SmartPtrp1(new?int(0));
????p1?=?p1;
????//?Test?Copy?Constructor
????SmartPtrp2(p1);
????//?Test?Assignment?Operator?Overloaded
????SmartPtrp3(new?int(1));
????p3?=?p1;
????
????return?0;
}

  測試結(jié)果如下:

  

參考資料

  《C++標(biāo)準(zhǔn)庫(第二版)》  

  C++中智能指針的設(shè)計(jì)和使用


本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動(dòng)力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉