智能指針

智能指針是一個(gè)類,用來(lái)存儲(chǔ)指向動(dòng)態(tài)分配對(duì)象的指針,負(fù)責(zé)自動(dòng)釋放動(dòng)態(tài)分配的對(duì)象吊圾,防止內(nèi)存泄漏。動(dòng)態(tài)分配的資源交給一個(gè)類對(duì)象去管理翰蠢,當(dāng)類對(duì)象聲明周期結(jié)束時(shí)项乒,自動(dòng)調(diào)用析構(gòu)函數(shù)釋放資源

shared_ptr

采用引用計(jì)數(shù)器的方法梁沧,允許多個(gè)智能指針指向同一個(gè)對(duì)象檀何,每當(dāng)多一個(gè)指針指向該對(duì)象時(shí),指向該對(duì)象的所有智能指針內(nèi)部的引用計(jì)數(shù)加1廷支,每當(dāng)減少一個(gè)智能指針指向?qū)ο髸r(shí)频鉴,引用計(jì)數(shù)減1,當(dāng)計(jì)數(shù)為0時(shí)自動(dòng)釋放動(dòng)態(tài)分配的資源恋拍。

#include <memory>
#include <string>
#include <iostream>

using namespace std;

int main(int argc, char** argv)
{
    string str = "hello world";
    shared_ptr<string> p1(new string(str));

    shared_ptr<string> p2(p1);

    shared_ptr<string> p3 = p1;             //允許多個(gè)指針指向同一個(gè)對(duì)象

    cout << *p1 << endl;
    cout << *p2 << endl;
    cout << *p3 << endl;

    shared_ptr<string> ps1;

    //從普通指針構(gòu)造智能指針
    string* str1 = new string("xian");

    //ps1 = str1;                       //隱式轉(zhuǎn)換不可以

    ps1 = shared_ptr<string>(str1);     //顯示轉(zhuǎn)換可以

    //shared_ptr<string> ps2 = str1;    //隱式轉(zhuǎn)換不可以

    //shared_ptr<string> ps3(str1);     //顯式轉(zhuǎn)換(出現(xiàn)程序崩潰垛孔,因?yàn)槎鄠€(gè)智能指針指向了同一塊內(nèi)存,會(huì)引起重復(fù)釋放內(nèi)存的問題),
    shared_ptr<string> ps3(ps1);        //多個(gè)智能指針指向同一塊內(nèi)存芝囤,但是使用智能指針是采用引用計(jì)數(shù)的原理似炎,只有當(dāng)引用計(jì)數(shù)為0時(shí)辛萍,才delete這塊內(nèi)存

    cout << *ps1 << endl;
    cout << *ps3 << endl;

    return 0;

}

auto_ptr

主要是為了解決“有異常拋出時(shí)發(fā)生內(nèi)存泄漏”的問題悯姊。因?yàn)榘l(fā)生異常而無(wú)法正常釋放內(nèi)存。auto_ptr有拷貝語(yǔ)義贩毕,拷貝后源對(duì)象變得無(wú)效悯许,程序崩潰,而unique_ptr沒有拷貝語(yǔ)義辉阶,提供了移動(dòng)語(yǔ)義(使用std::move)進(jìn)行轉(zhuǎn)移先壕。所以unique_ptr比auto_ptr更安全瘩扼。

#include <iostream>

using namespace std;

int main(int argc, char** argv)
{
    string* str1 = new string("hello ");
    auto_ptr<string> p1(str1);

    auto_ptr<string> p2(p1);                //自動(dòng)將p1的所有權(quán)剝奪,導(dǎo)致p1無(wú)法訪問垃僚,不夠安全集绰,建議使用unique_ptr

    //cout << *p1 << endl;

    cout << *p2 << endl;

    return 0;
}

unique_ptr

采用獨(dú)享所有權(quán)定義,一個(gè)非空的unique_ptr總是擁有它所指向的資源谆棺,轉(zhuǎn)移一個(gè)unique_ptr會(huì)把所有權(quán)全部從源指針轉(zhuǎn)移給目標(biāo)指針栽燕,源指針被置空;所以unique_ptr不支持普通的拷貝和賦值操作改淑。

#include <iostream>

using namespace std;

int main(int argc, char** argv)
{
    string* str1 = new string("hello ");
    unique_ptr<string> p1(str1);

    //unique_ptr<string> p2 = p1;       //不允許這樣做

    unique_ptr<string> p2 = move(p1);

    //cout << *p1 << endl;              //p1已經(jīng)喪失了所有權(quán)
    cout << *p2 << endl;

    return 0;
}

weak_ptr

弱引用碍岔。引用計(jì)數(shù)有一個(gè)問題就是互相引用形成環(huán)(環(huán)形引用),這樣兩個(gè)指針指向的內(nèi)存都無(wú)法釋放朵夏。需要使用weak_ptr打破環(huán)形應(yīng)用蔼啦,是為了配合shared_ptr而引入的一種智能指針,它指向一個(gè)由shared_ptr管理的對(duì)象而不影響所指對(duì)象的生命周期仰猖,也就是說(shuō)它只引用捏肢,不計(jì)數(shù)

#include <iostream>

using namespace std;

class A;
class B;

class A
{
public:

    //shared_ptr<B> b_ptr;
    weak_ptr<B> b_ptr;                  //修改為若應(yīng)用即可解決循環(huán)引用的問題
};

class B
{
public:

    //shared_ptr<A> a_ptr;
    weak_ptr<A> a_ptr;                  //修改為若應(yīng)用即可解決循環(huán)引用的問題
};

int main(int argc, char** argv)
{

    string* str1 = new string("hell0");

    shared_ptr<string> p(str1);

    //weak_ptr<string> p1(str1);                //出錯(cuò)

    weak_ptr<string> p1(p);                     //可以指向已有的shared_ptr

    //cout << *p1 << endl;                      //不支持*操作
    cout << p1.use_count() << endl;             //查看與自己指向同一個(gè)對(duì)象的shared_ptr的引用計(jì)數(shù)饥侵,但是自己只引用不會(huì)增加或減少計(jì)數(shù)

    shared_ptr<string> p2 = p1.lock();
    cout << *p2 << endl;                        //借助lock函數(shù)返回一個(gè)和p1指向同一個(gè)對(duì)象的shared_ptr類型指針猛计,獲取其存儲(chǔ)的數(shù)據(jù)。
    cout << p1.use_count() << endl;

    //**************循環(huán)引用******************
    shared_ptr<A> pa(new A);
    shared_ptr<B> pb(new B);

    pa->b_ptr = pb;
    pb->a_ptr = pa;

    //如果不是弱引用爆捞,兩個(gè)的引用計(jì)數(shù)都為2奉瘤,這個(gè)情況下最后的引用計(jì)數(shù)不為2,造成兩個(gè)內(nèi)存得不到釋放煮甥,導(dǎo)致內(nèi)存泄漏
    cout << "pa.use_count=" << pa.use_count() << endl;
    cout << "pb.use_count=" << pb.use_count() << endl;  

    return 0;
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末盗温,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子成肘,更是在濱河造成了極大的恐慌卖局,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,755評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件双霍,死亡現(xiàn)場(chǎng)離奇詭異砚偶,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)洒闸,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門染坯,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人丘逸,你說(shuō)我怎么就攤上這事单鹿。” “怎么了深纲?”我有些...
    開封第一講書人閱讀 165,138評(píng)論 0 355
  • 文/不壞的土叔 我叫張陵仲锄,是天一觀的道長(zhǎng)劲妙。 經(jīng)常有香客問我,道長(zhǎng)儒喊,這世上最難降的妖魔是什么镣奋? 我笑而不...
    開封第一講書人閱讀 58,791評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮怀愧,結(jié)果婚禮上唆途,老公的妹妹穿的比我還像新娘。我一直安慰自己掸驱,他們只是感情好肛搬,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,794評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著毕贼,像睡著了一般温赔。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上鬼癣,一...
    開封第一講書人閱讀 51,631評(píng)論 1 305
  • 那天陶贼,我揣著相機(jī)與錄音,去河邊找鬼待秃。 笑死拜秧,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的章郁。 我是一名探鬼主播枉氮,決...
    沈念sama閱讀 40,362評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼暖庄!你這毒婦竟也來(lái)了聊替?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,264評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤培廓,失蹤者是張志新(化名)和其女友劉穎惹悄,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體肩钠,經(jīng)...
    沈念sama閱讀 45,724評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡泣港,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了价匠。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片当纱。...
    茶點(diǎn)故事閱讀 40,040評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖霞怀,靈堂內(nèi)的尸體忽然破棺而出惫东,到底是詐尸還是另有隱情莉给,我是刑警寧澤毙石,帶...
    沈念sama閱讀 35,742評(píng)論 5 346
  • 正文 年R本政府宣布廉沮,位于F島的核電站,受9級(jí)特大地震影響徐矩,放射性物質(zhì)發(fā)生泄漏滞时。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,364評(píng)論 3 330
  • 文/蒙蒙 一滤灯、第九天 我趴在偏房一處隱蔽的房頂上張望坪稽。 院中可真熱鬧,春花似錦鳞骤、人聲如沸窒百。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)篙梢。三九已至,卻和暖如春美旧,著一層夾襖步出監(jiān)牢的瞬間渤滞,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工榴嗅, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留妄呕,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,247評(píng)論 3 371
  • 正文 我出身青樓嗽测,卻偏偏與公主長(zhǎng)得像绪励,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子唠粥,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,979評(píng)論 2 355