亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

詳解C++中shared_ptr的使用教程

2020-01-26 14:34:42
字體:
來源:轉載
供稿:網友

shared_ptr是一種智能指針(smart pointer)。shared_ptr的作用有如同指針,但會記錄有多少個shared_ptrs共同指向一個對象。
這便是所謂的引用計數(reference counting)。一旦最后一個這樣的指針被銷毀,也就是一旦某個對象的引用計數變為0,這個對象會被自動刪除。這在非環形數據結構中防止資源泄露很有幫助。
auto_ptr由于它的破壞性復制語義,無法滿足標準容器對元素的要求,因而不能放在標準容器中;如果我們希望當容器析構時能自動把它容納的指針元素所指的對象刪除時,通常采用一些間接的方式來實現,顯得比較繁瑣。boost庫中提供了一種新型的智能指針shared_ptr,它解決了在多個指針間共享對象所有權的問題,同時也滿足容器對元素的要求,因而可以安全地放入容器中。

總結下幾個使用shared_ptr需要注意的問題:

一. 相互引用鏈

class C;class B : public std::enable_shared_from_this<B>{public: ~B(){ cout << "~B" << endl; } void SetPC(std::shared_ptr<C>& pc){ _pc = pc; } private: std::shared_ptr<C> _pc;};class C : public std::enable_shared_from_this<C>{public: ~C(){ cout << "~C" << endl; } void SetPB(std::shared_ptr<B>& pb){ _pb = pb; } private: std::shared_ptr<B> _pb;};int main(){ std::shared_ptr<C> pc = std::make_shared<C>(); std::shared_ptr<B> pb = std::make_shared<B>(); pc->SetPB(pb); pb->SetPC(pc); return 0;}

上面的代碼中,B和C均不能正確析構,正確的做法是,在B和C的釋放函數,如Close中,將其包含的shared_ptr置空。這樣才能解開引用鏈。

二. 自引用
還有個比較有意思的例子:

class C : public std::enable_shared_from_this < C >{public: ~C() {  std::cout << "~C" << std::endl; } int32_t Decode(const char* data, size_t) {  return 0; } void SetDecoder(std::function<int32_t(const char*, size_t)> decoder) {  _decoder = decoder; }private: std::function<int32_t(const char*, size_t)> _decoder;};int main(){ {  std::shared_ptr<C> pc = std::make_shared<C>();  auto decoder = std::bind(&C::Decode, pc, std::placeholders::_1, std::placeholders::_2);  pc->SetDecoder(decoder); } // C不能正確析構 因為存在自引用 return 0;}

上面的C類包含了一個function,該function通過std::bind引用了一個std::shared_ptr,所以_decoder其實包含了一個對shared_ptr的引用。導致C自引用了自身,不能正確析構。需要在C的Close之類的執行關閉函數中,將_decoder=nullptr,以解開這種自引用。

三. 類中傳遞
下面的例子中有個更為隱蔽的問題:

class Session : public std::enable_shared_from_this < Session >{public: ~Session() {  std::cout << "~C" << std::endl; } void Start() {  // 進行一些異步調用  // 如 _socket.async_connect(..., boost::bind(&Session::ConnectCompleted, this), boost::asio::placeholders::error, ...) } void ConnectCompleted(const boost::system::err_code& err) { if(err) return;   // ... 進行處理  // 如 _socket.async_read(..., boost::bind(&Session::ReadCompleted, this), boost::asio::placeholders::error, ...) } void Session::ReadComplete(const boost::system::error_code& err, size_t bytes_transferred) {  if (err || bytes_transferred == 0)  {   DisConnect();   return;  } // 處理數據 繼續讀 // ProcessData(); // _socket.async_read(...) }private: std::function<int32_t(const char*, size_t)> _decoder;};int main(){ {  std::shared_ptr<Session> pc = std::make_shared<Session>();  pc->Start(); } return 0;}

上面Session,在調用Start時,調用了異步函數,并回調自身,如果在回調函數的 boost::bind 中 傳入的是shared_from_this(),那么并無問題,shared_ptr將被一直傳遞下去,在網絡處理正常時,Session將正常運行,即使main函數中已經沒有它的引用,但是它靠boost::bind”活了下來”,boost::bind會保存傳給它的shared_ptr,在調用函數時傳入。當網絡遇到錯誤時,函數直接返回。此時不再有新的bind為其”續命”。Session將被析構。
而真正的問題在于,如果在整個bind鏈中,直接傳遞了this指針而不是shared_from_this(),那么實際上當函數執行完成后,Session即會析構,包括其內部的資源(如 _socket)也會被釋放。那么當boost底層去執行網絡IO時,自然會遇到錯誤,并且仍然會”正?!被卣{到對應函數,如ReadCompleted,然后在err中告訴你:”由本地系統終止網絡連接”(或:”An attempt to abort the evaluation failed. The process is now in an indeterminate state.” )。讓人誤以為是網絡問題,很難調試。而事實上此時整個對象都已經被釋放掉了。
注:由于C++對象模型實現所致,成員函數和普通函數的主要區別如下:

  • 成員函數帶隱式this參數
  • 成員函數具有訪問作用域,并且函數內會對非靜態成員變量訪問做一些轉換,如 _member_data 轉換成 this->_member_data;

也就是說,成員函數并不屬于對象,非靜態數據成員才屬于對象。

因此如下調用在編譯期是合法的:

((A*)nullptr)->Func();

而如果成員函數A::Func()沒有訪問A的非靜態成員變量,這段代碼甚至能正確運行,如:

class Test{public: void Say() {  std::cout << "Say Test" << std::endl; } void Set(int data) {  _data = data; }private: int _data;};int main(){ // 運行成功 ((Test*)nullptr)->Say(); // 運行會崩掉,嘗試訪問空指針所指內存(_data) ((Test*)nullptr)->Set(1); return 0;}

正因為這種特性,有時候在成員函數中糾結半天,也不會注意到這個對象已經”不正常了”,被釋放掉了。

四. shared_ptr 使用總結
盡量不要環引用或自引用,可通過weak_ptr來避免環引用:owner持有child的shared_ptr child持有owner的weak_ptr
如果存在環引用或自引用,記得在釋放時解開這個引用鏈
對于通過智能指針管理的類,在類中通過shared_from_this()而不是this來傳遞本身
在類釋放時,盡量手動置空其所有的shared_ptr成員,包括function

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美日韩国产在线看| 中文字幕欧美日韩va免费视频| 国产精品777| 亚洲国产一区自拍| 九九视频直播综合网| 欧美日韩激情视频8区| 国产一区二中文字幕在线看| 日韩在线观看高清| 国产自产女人91一区在线观看| 中文字幕视频在线免费欧美日韩综合在线看| 国产精品欧美一区二区| 爱福利视频一区| 精品国偷自产在线视频99| 久久激情视频久久| 国产亚洲美女精品久久久| 亚洲精品国产精品国产自| 成人精品在线视频| 青青久久av北条麻妃黑人| 久久久久亚洲精品国产| 精品国产鲁一鲁一区二区张丽| 久久欧美在线电影| 亚洲精品网站在线播放gif| 欧美www视频在线观看| 久久夜色精品国产亚洲aⅴ| 久久99亚洲热视| 国产精品入口福利| 国产精品一区二区在线| 国产精彩精品视频| 伊人男人综合视频网| 欧美在线一区二区视频| 国产精品网站大全| 久久久久成人网| 成人动漫网站在线观看| 国产日韩综合一区二区性色av| 亚洲欧美激情视频| 欧美日韩亚洲天堂| 日韩av影片在线观看| 综合欧美国产视频二区| 国产做受高潮69| 日韩精品高清在线观看| 久久人人爽人人爽人人片av高清| 国产精品久久在线观看| 久热99视频在线观看| 亚洲欧美另类自拍| 精品国产91久久久| 色播久久人人爽人人爽人人片视av| 俺去啦;欧美日韩| 亚洲电影中文字幕| 性金发美女69hd大尺寸| 亚洲欧美另类国产| 国产福利视频一区二区| 亚洲第一区中文99精品| 国产精品亚洲аv天堂网| 亚洲欧美国产高清va在线播| 91精品国产乱码久久久久久蜜臀| 欧美激情国内偷拍| 神马久久桃色视频| 久久中文字幕一区| 久久视频在线免费观看| 亚洲精品在线视频| 欧美肥臀大乳一区二区免费视频| 日韩激情av在线播放| 日韩小视频在线观看| 成人欧美一区二区三区黑人| 中文字幕av一区二区三区谷原希美| 日韩国产精品视频| 色综合久久天天综线观看| 亚洲人成啪啪网站| 一本一本久久a久久精品综合小说| 欧美疯狂性受xxxxx另类| 国产亚洲精品久久久久久777| 一二美女精品欧洲| 欧美电影免费播放| 欧美电影免费在线观看| 久久久久久久久久久免费| 成人网欧美在线视频| 国产主播喷水一区二区| 91香蕉国产在线观看| 2019中文字幕在线观看| 亚洲天堂免费视频| 亚洲成人网在线| 国产在线日韩在线| 97视频com| 国产成人欧美在线观看| 91精品免费久久久久久久久| 精品久久久中文| 97福利一区二区| 国产精品免费久久久久影院| 成人激情黄色网| 日韩经典一区二区三区| 78色国产精品| 欧美日韩在线一区| 日韩av在线免播放器| 国产婷婷97碰碰久久人人蜜臀| 欧美一区二区大胆人体摄影专业网站| 中日韩美女免费视频网址在线观看| 岛国av一区二区在线在线观看| 国产视频亚洲精品| 欧美大片免费观看在线观看网站推荐| 亚洲国产欧美精品| 国产精品网站视频| 国产一区二区欧美日韩| 在线观看久久av| 色一情一乱一区二区| 青青草精品毛片| 国产91精品黑色丝袜高跟鞋| 亚洲已满18点击进入在线看片| 成人免费淫片aa视频免费| 久久久久久久影院| 日韩av免费在线看| 日韩免费在线播放| 午夜精品99久久免费| 国内精品久久久久久久| 久久久久久久久久久网站| 久久国产色av| 日韩高清不卡av| 奇米四色中文综合久久| 成人美女av在线直播| 久久人人爽人人| 国产精品日韩精品| 色噜噜亚洲精品中文字幕| 日韩av电影院| 亚洲小视频在线| 亚洲aⅴ日韩av电影在线观看| 精品视频在线播放| 亚洲日本成人女熟在线观看| 亚洲天堂精品在线| 丝袜亚洲欧美日韩综合| 欧美日韩加勒比精品一区| 日本精品免费观看| 日韩大胆人体377p| 日韩电影大全免费观看2023年上| 国产精品无码专区在线观看| 亚洲视频欧美视频| 久久久久亚洲精品成人网小说| 亚洲另类激情图| 国产一区二区三区丝袜| 亚洲精品www久久久久久广东| 欧美精品制服第一页| 日韩有码在线观看| 日韩av电影在线网| 国产精品视频专区| 亚洲精品欧美一区二区三区| 国产精品黄色影片导航在线观看| 欧美做受高潮电影o| 国产精品国产自产拍高清av水多| 国产日韩专区在线| 国产成人在线精品| 成人精品在线观看| 国产亚洲成av人片在线观看桃| 一区二区三区视频观看| 在线成人激情视频| 国产乱肥老妇国产一区二| 国内精品久久久久久中文字幕| 国产国语刺激对白av不卡| 精品国内产的精品视频在线观看| 亚洲精品自拍视频| 亚洲成人国产精品| 97**国产露脸精品国产| 国产亚洲成精品久久| 久久精品成人一区二区三区| 欧美国产亚洲精品久久久8v| 一区二区三区久久精品| 国外成人免费在线播放|