日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

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

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

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

使用智能指針的緣由

1. 考慮下邊的簡(jiǎn)單代碼:


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

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

2. 考慮另一簡(jiǎn)單代碼:


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

  我們可能會(huì)心想,這下程序應(yīng)該沒(méi)問(wèn)題了?可實(shí)際上程序還是有問(wèn)題。上邊程序雖然最后釋放了申請(qǐng)的內(nèi)存,但ptr會(huì)變成空懸指針(dangling pointer,也就是野指針)。空懸指針不同于空指針(nullptr),它會(huì)指向“垃圾”內(nèi)存,給程序帶去諸多隱患(如我們無(wú)法用if語(yǔ)句來(lái)判斷野指針)。

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


ptr?=?nullptr;

  除了上邊考慮到的兩個(gè)問(wèn)題,上邊程序還存在另一問(wèn)題:如果內(nèi)存申請(qǐng)不成功,new會(huì)拋出異常,而我們卻什么都沒(méi)有做!所以對(duì)這程序我們還得繼續(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. 考慮最后一簡(jiǎn)單代碼:


#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)致程序終止,而已申請(qǐng)的內(nèi)存并沒(méi)有釋放掉。

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


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

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

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

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

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

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

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

智能指針

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

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

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

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

shared_ptr

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

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

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

  下邊程序摘自《C++標(biāo)準(zhǔn)庫(kù)(第二版)》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)于程序邏輯可見(jiàn)下圖:

  

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

  1)對(duì)智能指針pNico的拷貝是淺拷貝,所以當(dāng)我們改變對(duì)象“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,不過(guò)只對(duì)“由new建立起來(lái)的單一對(duì)象”起作用。當(dāng)然我們也可以自己定義內(nèi)存釋放器,就如上述程序。不過(guò)值得注意的是,默認(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)庫(kù)自C++11起開(kāi)始提供的類(lèi)型。它是一種在異常發(fā)生時(shí)可幫助避免資源泄露的智能指針。一般而言,這個(gè)智能指針實(shí)現(xiàn)了獨(dú)占式擁有概念,意味著它可確保一個(gè)對(duì)象和其相應(yīng)資源同一時(shí)間只被一個(gè)指針擁有。一旦擁有者被銷(xiāo)毀或變成空,或開(kāi)始擁有另一個(gè)對(duì)象,先前擁有的那個(gè)對(duì)象就會(huì)被銷(xiāo)毀,其任何相應(yīng)資源也會(huì)被釋放。

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


#includeusing?namespace?std;

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

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

  基于引用計(jì)數(shù)的智能指針可以簡(jiǎn)單實(shí)現(xiàn)如下(詳細(xì)解釋見(jiàn)程序中注釋?zhuān)?/p>


#includeusing?namespace?std;

templateclass?SmartPtr
{
public:
????SmartPtr(T?*p);
????~SmartPtr();
????SmartPtr(const?SmartPtr&orig);????????????????//?淺拷貝
????SmartPtr&?operator=(const?SmartPtr&rhs);????//?淺拷貝
private:
????T?*ptr;
????//?將use_count聲明成指針是為了方便對(duì)其的遞增或遞減操作
????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è)對(duì)象引用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;
}

//?重載等號(hào)函數(shù)不同于復(fù)制構(gòu)造函數(shù),即等號(hào)左邊的對(duì)象可能已經(jīng)指向某塊內(nèi)存。
//?這樣,我們就得先判斷左邊對(duì)象指向的內(nèi)存已經(jīng)被引用的次數(shù)。如果次數(shù)為1,
//?表明我們可以釋放這塊內(nèi)存;反之則不釋放,由其他對(duì)象來(lá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ù)對(duì)象的使用計(jì)數(shù)減1,若該對(duì)象的使用計(jì)數(shù)減至0,則刪除該對(duì)象
????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;
}

  測(cè)試程序如下:


#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;
}

  測(cè)試結(jié)果如下:

  

參考資料

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

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


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

LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動(dòng)電源

在工業(yè)自動(dòng)化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動(dòng)力設(shè)備,其驅(qū)動(dòng)電源的性能直接關(guān)系到整個(gè)系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動(dòng)勢(shì)抑制與過(guò)流保護(hù)是驅(qū)動(dòng)電源設(shè)計(jì)中至關(guān)重要的兩個(gè)環(huán)節(jié),集成化方案的設(shè)計(jì)成為提升電機(jī)驅(qū)動(dòng)性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動(dòng)電源

LED 驅(qū)動(dòng)電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個(gè)照明設(shè)備的使用壽命。然而,在實(shí)際應(yīng)用中,LED 驅(qū)動(dòng)電源易損壞的問(wèn)題卻十分常見(jiàn),不僅增加了維護(hù)成本,還影響了用戶(hù)體驗(yàn)。要解決這一問(wèn)題,需從設(shè)計(jì)、生...

關(guān)鍵字: 驅(qū)動(dòng)電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動(dòng)電源的公式,電感內(nèi)電流波動(dòng)大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計(jì) 驅(qū)動(dòng)電源

電動(dòng)汽車(chē)(EV)作為新能源汽車(chē)的重要代表,正逐漸成為全球汽車(chē)產(chǎn)業(yè)的重要發(fā)展方向。電動(dòng)汽車(chē)的核心技術(shù)之一是電機(jī)驅(qū)動(dòng)控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動(dòng)系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動(dòng)汽車(chē)的動(dòng)力性能和...

關(guān)鍵字: 電動(dòng)汽車(chē) 新能源 驅(qū)動(dòng)電源

在現(xiàn)代城市建設(shè)中,街道及停車(chē)場(chǎng)照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨(dú)特的優(yōu)勢(shì)逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動(dòng)電源 LED

LED通用照明設(shè)計(jì)工程師會(huì)遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動(dòng)電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動(dòng)電源的電磁干擾(EMI)問(wèn)題成為了一個(gè)不可忽視的挑戰(zhàn)。電磁干擾不僅會(huì)影響LED燈具的正常工作,還可能對(duì)周?chē)娮釉O(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來(lái)解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動(dòng)電源

開(kāi)關(guān)電源具有效率高的特性,而且開(kāi)關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動(dòng)電源

關(guān)鍵字: LED 驅(qū)動(dòng)電源 開(kāi)關(guān)電源

LED驅(qū)動(dòng)電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動(dòng)LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動(dòng)電源
關(guān)閉