C++ shared_ptr相關(guān)技術(shù)

概 述

整理一下 shared_ptr 的相關(guān)技術(shù)以及其使用的注意項缓屠。

整 理

1. 用make_shared創(chuàng)建智能指針
shared_ptr 內(nèi)部包含一個托管對象的原始指針以及一個引用計數(shù)癞揉,因此直接使用 new 來創(chuàng)建一個 shared_ptr 需要兩次內(nèi)存分配:一個用于托管對象复隆,另一個用于引用計數(shù),并且兩塊內(nèi)存空間是不連續(xù)的吞获。而make_shared只需要分配一次內(nèi)存渔嚷,并在其中創(chuàng)建兩個,并且沒有暴露指向托管對象的原始指針:

// std::shared_ptr<int> p(new int);  改為:
std::shared_ptr<int> p = std::make_shared<int>();

因此make_shared更高效媳握,對 CPU 緩存友好碱屁,且避免了內(nèi)存泄漏的任何可能性。
缺點
延遲了內(nèi)存釋放的時間:由于make_shared只分配一次內(nèi)存蛾找,所以將對象的生命周期與控制塊(強引用娩脾、弱引用的信息)的生命周期捆綁到一起,原本強引用計數(shù)為零時就可以釋放內(nèi)存打毛,現(xiàn)在變?yōu)榱藦娨檬辽蕖⑷跻枚紴榱銜r才能釋放,延遲了內(nèi)存釋放的時間

2. 減少 shared_pt 的拷貝
正如上文所說幻枉,shared_ptr 的內(nèi)存占用是原始指針的兩倍碰声,因此 shared_ptr 的拷貝開銷比拷貝原始指針要高。

  • 使用std::move()來移動 shared_ptr 的所有權(quán)
// 當(dāng)一個 shared_ptr 需要將其所有權(quán)轉(zhuǎn)移給另外一個新的 shared_ptr 時
std::shared_ptr<int> p1 = std::make_shared<int>();
// 使用 std::move()熬甫,減少拷貝
std::shared_ptr<int> p2 = std::move(p1); //此時 p1 等于 nullptr !
  • 以 const reference 方式傳遞
void add(const shared_ptr<number>& pNumber);
void subtract(const shared_ptr<number>& pNumber);

// 傳常引用(pass by const reference)胰挑,減少拷貝
void func(const int& num) 
{
    auto pNumber = std::make_shared<number>(num);
    add(pNumber);
    subtract(pNumber);
}

3. enable_shared_from_this
當(dāng)我們使用異步回調(diào)時,通常會需要傳入當(dāng)前類對象椿肩,或者使用類成員函數(shù)作為回調(diào)瞻颂。例如:

class obj 
{
    // ...
};

void obj::Init()
{
    m_pAsyncTimer->SetCallback(std::bind(&obj::Func, this, std::placeholders::_1));
};

如果直接使用this來傳入,無法保證在回調(diào)函數(shù)中操作的this對象依然有效郑象。
對此贡这,我們可以使用enable_shared_from_this來使得當(dāng)前對象this變?yōu)?shared_ptr,從而延長對象的生命周期厂榛,保證在異步回調(diào)時盖矫,對象依然有效丽惭。

class obj : public std::enable_shared_from_this<obj>  
{
    // ...
};

void obj::Init()
{
    m_pAsyncTimer->SetCallback(std::bind(&obj::Func, shared_from_this(), std::placeholders::_1));
};

注:使用shared_from_this()需要對象在堆上創(chuàng)建,并且由 shared_ptr 管理其生命周期炼彪。因此shared_from_this()不能在構(gòu)造函數(shù)中調(diào)用吐根,因為在構(gòu)造對象時,其還沒有交給 shared_ptr 接管辐马。
此外拷橘,對于在類的成員函數(shù)中需要把當(dāng)前類對象作為參數(shù)傳遞給其他函數(shù)時,如果直接傳遞this喜爷,當(dāng)前對象就會被多個 shared_ptr 管理冗疮,造成二次釋放的錯誤。使用enable_shared_from_this配合shared_from_this()也可以很好的解決此問題檩帐。

4. 弱回調(diào)
上文提到的shared_from_this()延長了對象的生命周期术幔,如果我們不想額外延長對象的生命周期。例如針對如果對象還活著湃密,就調(diào)用它的回調(diào)函數(shù)诅挑,反之忽略這樣的需求,我們可以使用 weak_ptr 來傳入std::function泛源,這樣就不會延長對象的生命周期拔妥。在回調(diào)時,嘗試提升為 shared_ptr达箍,如果成功没龙,說明對象有效,那么執(zhí)行回調(diào)缎玫;反之則忽略硬纤。

5. 自定義析構(gòu)
shared_ptr 的構(gòu)造函數(shù)允許傳入自定義刪除器Deleter

template< class Y, class Deleter >
shared_ptr( Y* ptr, Deleter d );

因此,shared_ptr 支持自定義析構(gòu)動作赃磨。在繼承關(guān)系中筝家,使用 shared_ptr 管理,虛析構(gòu)不再是必須的邻辉。
需要注意的是肛鹏,上文提到的make_shared構(gòu)造智能指針不允許自定義刪除器,刪除器Deleter支持std::function對象

#include <memory>
#include <functional>

class Sample {
public:
    // ...
};

void deleter(Sample* x) {
    // ...
}

int main() {
    std::shared_ptr<Sample> p1(new Sample, deleter);

    std::shared_ptr<Sample> p2(new Sample, [](Sample* x) {
            // ...
        });

    std::shared_ptr<Sample> p3(new Sample, std::bind(deleter, std::placeholders::_1));

    return 0;
}

6. 析構(gòu)所在的線程
對象的析構(gòu)是同步的恩沛,當(dāng)指向?qū)ο蟮淖詈笠粋€ shared_ptr 離開作用域時,對象就會在當(dāng)前線程析構(gòu)缕减。這有可能會影響到關(guān)鍵線程的速度雷客,對此可以用一個單獨的線程來專門做析構(gòu)。
由于 shared_ptr<void> 可以持有任何對象并安全的釋放桥狡,我們可以通過一個BlockingQueue<shared_ptr<void>>(shared_ptr<void>的阻塞隊列)把對象的析構(gòu)都轉(zhuǎn)移到專用線程搅裙,從而解放關(guān)鍵線程皱卓。

7. 循環(huán)引用
若A持有B的 shared_ptr,且B也持有A的 shared_ptr部逮,此時會造成循環(huán)引用娜汁,導(dǎo)致A、B都無法釋放兄朋。此時掐禁,我們可以使用weak_ptr
weak_ptr不會增加引用計數(shù)颅和,因此可以打破 shared_ptr 的循環(huán)引用傅事。通常在繼承關(guān)系中的做法是父類持有子類的 shared_ptr,子類持有指向父類的 weak_ptr峡扩。

參考:
《Linux多線程服務(wù)端編程》陳碩
https://stackoverflow.com/questions/18301511
https://stackoverflow.com/questions/712279
https://zh.cppreference.com/w/cpp/memory/shared_ptr/make_shared

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末蹭越,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子教届,更是在濱河造成了極大的恐慌响鹃,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,734評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件案训,死亡現(xiàn)場離奇詭異买置,居然都是意外死亡,警方通過查閱死者的電腦和手機萤衰,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評論 3 394
  • 文/潘曉璐 我一進店門堕义,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人脆栋,你說我怎么就攤上這事倦卖。” “怎么了椿争?”我有些...
    開封第一講書人閱讀 164,133評論 0 354
  • 文/不壞的土叔 我叫張陵怕膛,是天一觀的道長。 經(jīng)常有香客問我秦踪,道長褐捻,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,532評論 1 293
  • 正文 為了忘掉前任椅邓,我火速辦了婚禮柠逞,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘景馁。我一直安慰自己板壮,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,585評論 6 392
  • 文/花漫 我一把揭開白布合住。 她就那樣靜靜地躺著绰精,像睡著了一般撒璧。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上笨使,一...
    開封第一講書人閱讀 51,462評論 1 302
  • 那天卿樱,我揣著相機與錄音,去河邊找鬼硫椰。 笑死繁调,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的最爬。 我是一名探鬼主播涉馁,決...
    沈念sama閱讀 40,262評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼爱致!你這毒婦竟也來了烤送?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,153評論 0 276
  • 序言:老撾萬榮一對情侶失蹤糠悯,失蹤者是張志新(化名)和其女友劉穎帮坚,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體互艾,經(jīng)...
    沈念sama閱讀 45,587評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡试和,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,792評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了纫普。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片阅悍。...
    茶點故事閱讀 39,919評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖昨稼,靈堂內(nèi)的尸體忽然破棺而出节视,到底是詐尸還是另有隱情,我是刑警寧澤假栓,帶...
    沈念sama閱讀 35,635評論 5 345
  • 正文 年R本政府宣布寻行,位于F島的核電站,受9級特大地震影響匾荆,放射性物質(zhì)發(fā)生泄漏拌蜘。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,237評論 3 329
  • 文/蒙蒙 一牙丽、第九天 我趴在偏房一處隱蔽的房頂上張望简卧。 院中可真熱鬧,春花似錦烤芦、人聲如沸举娩。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽晓铆。三九已至,卻和暖如春绰播,著一層夾襖步出監(jiān)牢的瞬間骄噪,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評論 1 269
  • 我被黑心中介騙來泰國打工蠢箩, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留链蕊,地道東北人。 一個月前我還...
    沈念sama閱讀 48,048評論 3 370
  • 正文 我出身青樓谬泌,卻偏偏與公主長得像滔韵,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子掌实,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,864評論 2 354

推薦閱讀更多精彩內(nèi)容