<menu id="w8yyk"><menu id="w8yyk"></menu></menu>
  • <dd id="w8yyk"><nav id="w8yyk"></nav></dd>
    <menu id="w8yyk"></menu>
    <menu id="w8yyk"><code id="w8yyk"></code></menu>
    <menu id="w8yyk"></menu>
    <xmp id="w8yyk">
    <xmp id="w8yyk"><nav id="w8yyk"></nav>
  • 網站首頁 > 物聯資訊 > 技術分享

    學習筆記:shared_ptr陷阱

    2016-09-28 00:00:00 廣州睿豐德信息科技有限公司 閱讀
    睿豐德科技 專注RFID識別技術和條碼識別技術與管理軟件的集成項目。質量追溯系統、MES系統、金蝶與條碼系統對接、用友與條碼系統對接 條款1:不要把一個原生指針給多個shared_ptr管理 int* ptr = new int; shared_ptr<int> p1(ptr); shared_ptr<int> p2(ptr); //logic error ptr對象被刪除了2次 這種問題比喻成“二龍治水”,在原生指針中也同樣可能發生。   條款2:不要把this指針給shared_ptr class Test{ public:     void Do(){  m_sp =  shared_ptr<Test>(this);  } private:     shared_ptr<Test> m_member_sp; };   Test* t = new Test; shared_ptr<Test> local_sp(t); p->Do();   發生什么事呢,t對象被刪除了2次! t對象給了local_sp管理,然后在m_sp =  shared_ptr<Test>(this)這句里又請了一尊神來管理t。 這就發生了條款1里“二龍治水”錯誤。   條款3:shared_ptr作為被保護的對象的成員時,小心因循環引用造成無法釋放資源。   對象需要相互協作,對象A需要知道對象B的地址,這樣才能給對象B發消息(或調用其方法)。 設計模式中有大量例子,一個對象中有其他對象的指針。現在把原生指針替換為shared_ptr.   假設a對象中含有一個shared_ptr<B>指向b對象;假設b對象中含有一個shared_ptr<A>指向a對象 并且a,b對象都是堆中分配的。很輕易就能與他們失去最后聯系。 考慮某個shared_ptr<A> local_a;是我們能最后一個看到a對象的共享智能指針,其use_count==2, 因為對象b中持有a的指針。所以當local_a說再見時,local_a只是把a對象的use_count改成1。 同理b對象。然后我們再也看不到a,b的影子了,他們就靜靜的躺在堆里,成為斷線的風箏。   解決方案是:Use weak_ptr to "break cycles."(boost文檔里寫的)或者顯示的清理   條款4:不要在函數實參里創建shared_ptr   function ( shared_ptr<int>(new int), g( ) );  //有缺陷 可能的過程是先new int,然后調g( ),g( )發生異常,shared_ptr<int>沒有創建,int內存泄露   shared_ptr<int> p(new int()); f(p, g());  //Boost推薦寫法   條款5:對象內部生成shared_ptr   前面說過,不能把this指針直接扔給shared_ptr. 但是沒有禁止在對象內部生成自己的shared_ptr   //這是Boost的例子改的。 class Y: public boost::enable_shared_from_this<Y> {     boost::shared_ptr<Y> GetSelf()     {         return shared_from_this();     } };   原理是這樣的。普通的(沒有繼承enable_shared_from_this)類T的shared_ptr<T> p(new T). p作為棧對象占8個字節,為了記錄(new T)對象的引用計數,p會在堆上分配16個字節以保存 引用計數等“智能信息”。share_ptr沒有“嵌入(intrusive)”到T對象,或者說T對象對share_ptr毫不知   情。Y對象則不同,Y對象已經被“嵌入”了一些share_ptr相關的信息,目的是為了找到“全局性”的 那16字節的本對象的“智能信息”。   原理說完了,就是陷阱 Y y; boost::shared_ptr<Y> p=  y.GetSelf(); //無知的代碼,y根本就不是new出來的   Y* y = new Y; boost::shared_ptr<Y> p=  y->GetSelf(); //似是而非,仍舊程序崩盤。 Boost文檔說,在調用shared_from_this()之前,必須存在一個正常途徑創建的shared_ptr   boost::shared_ptr<Y> spy(new Y) boost::shared_ptr<Y> p =  spy->GetSelf(); //OK   條款6 :處理不是new的對象要小心。   int* pi = (int*)malloc(4) shared_ptr<int> sp( pi ) ; //delete馬嘴不對malloc驢頭。   條款7:多線程對引用計數的影響。   如果是輕量級的鎖,比如InterLockIncrement等,對程序影響不大 如果是重量級的鎖,就要考慮因為share_ptr維護引用計數而造成的上下文切換開銷。 1.33版本以后的shared_ptr對引用計數的操作使用的是Lock-Free(類似InterLockIncrement函數族) 的操作,應該效率不錯,而且能保證線程安全(庫必須保證其安全,程序員都沒有干預這些隱藏事物的機會)。 Boost文檔說read,write同時對shared_ptr操作時,行為不確定。這是因為shared_ptr本身有兩個成員px,pi。 多線程同時對px讀寫是要出問題的。與一個int的全局變量多線程讀寫會出問題的原因一樣。   條款8:對象數組用shared_array   int* pint = new int[100]; shared_array<int> p (pint );   既然shared_ptr對應著delete;顯然需要一個delete[]對應物shared_array   條款9:學會用刪除器   struct Test_Deleter {        void  operator ()( Test* p){   ::free(p);   } }; Test* t = (Test*)malloc(sizeof(Test)); new (t) Test;   shared_ptr<Test> sp( t ,  Test_Deleter() ); //刪除器可以改變share_ptr銷毀對象行為   有了刪除器,shared_array無用武之地了。 template<class T> struct Array_Deleter {        void  operator ()( T*){   delete[] p;   } }; int* pint = new int[100]; shared_ptr<int> p (pint, Array_Deleter<int>() );   條款10:學會用分配器   存放引用計數的地方是堆內存,需要16-20字節的開銷。 如果大量使用shared_ptr會造成大量內存碎片。 shared_ptr構造函數的第3個參數是分配器,可以解決這個問題。   shared_ptr<Test> p( (new Test), Test_Deleter(), Mallocator<Test>() ); 注意刪除器Test_Deleter是針對Test類的。分配器是針對shared_ptr內部數據的。   Mallocator<Test>()是個臨時對象(無狀態的),符合STL分配器規約。   template <typename T>  class Mallocator {      //略。。。。。。     T * allocate(const size_t n) const {         return singleton_pool<T,sizeof(T)>::malloc();     }     //略。。。。。。   Mallocator傳入Test,實際分配的類型確是 class boost::detail::sp_counted_impl_pda<class Test *,                                          struct Test_Deleter,                                          class Mallocator<class Test> > 這是用typeid(T).name()打印出來的。可能和rebind相關。   條款11 weak_ptr在使用前需要檢查合法性。 weak_ptr<K> wp; { shared_ptr<K>  sp(new K);  //sp.use_count()==1 wp = sp; //wp不會改變引用計數,所以sp.use_count()==1 shared_ptr<K> sp_ok = wp.lock(); //wp沒有重載->操作符。只能這樣取所指向的對象 } shared_ptr<K> sp_null = wp.lock(); //sp_null .use_count()==0; 因為上述代碼中sp和sp_ok離開了作用域,其容納的K對象已經被釋放了。 得到了一個容納NULL指針的sp_null對象。在使用wp前需要調用wp.expired()函數判斷一下。 因為wp還仍舊存在,雖然引用計數等于0,仍有某處“全局”性的存儲塊保存著這個計數信息。 直到最后一個weak_ptr對象被析構,這塊“堆”存儲塊才能被回收。否則weak_ptr無法直到自己 所容納的那個指針資源的當前狀態。   條款12 不要new shared_ptr<T>   本來shared_ptr就是為了管理指針資源的,不要又引入一個需要管理的指針資源shared_ptr<T>*   條款13  盡量不要get   class B{...}; class D : public B{ ...};  //繼承層次關系   shared_ptr<B> sp (new D);    //通過隱式轉換,儲存D的指針。 B* b = sp.get();             //shared_ptr辛辛苦苦隱藏的原生指針就這么被刨出來了。 D* d = dynamic_cast<D*>(b);  //這是使用get的正當理由嗎?   正確的做法 shared_ptr<B> spb (new D)  ; shared_ptr<D> spd = shared_dynamic_cast<D>(spb); //變成子類的指針 shared_ptr在竭盡全力表演的像一個原生指針,原生指針能干的事,它也基本上能干。   另一個同get相關的錯誤 shared_ptr<T> sp(new T); shared_ptr<T> sp2( sp.get() ) ;//又一個“二龍治水”實例,指針會刪2次而錯誤。   條款14 不要memcpy shared_ptr   shared_ptr<B> sp1 (new B)  ; shared_ptr<B> sp2; memcpy(&sp2,&sp1,sizeof(shared_ptr<B>)); //sp2.use_count()==1 很顯然,不是通過正常途徑(拷貝構造,賦值運算),引用計數是不會正確增長的。   條款15 使用BOOST預定義的宏去改變shared_ptr行為。   shared_ptr行為由類似BOOST_SP_DISABLE_THREADS這樣的宏控制。需要去學習他們到底是干什么的。 大師Andrei Alexandrescu設計了一種基于模板策略設計模式的智能指針,通過幾個模板參數去定制化 智能指針的行為。Boost卻不以為然,官方解釋是:需要統一的接口,這樣利于大規模書寫。 smart_ptr<T,OwnershipPolicy,ConversionPolicy,CheckingPolicy,StoragePolicy> sp(new T); 上述接口缺點是外形復雜,看上去像個大花臉。優點是客戶程序員可以輕易的定制行為。   條款17 構造函數里調用shared_from_this拋例外   class Holder:public enable_shared_from_this<Holder>{ public:     Holder() {         shared_ptr<Holder> sp = shared_from_this();         int x = sp.use_count();     } }; 同前面條款5,不符合enable_shared_from_this使用前提。   總結: 學習了一天就總結出10多條條款,長期研究一下恐怕就出現條款100了。為什么還要使用shared_ptr呢? 有很多開源庫用shared_ptr,而且shared_ptr具有“傳染性”(某網友語:像毒品沾上就甩不掉), 拋開它就會有更嚴重的多龍治水現象。shared_ptr作為原生指針的替代品,能解決一定的內存泄露問題。 實際上初學原生指針時,每個人都遇到過野指針,刪兩次,忘記刪除等問題。學習shared_ptr也會遇到。 shared_ptr的確能改善上述問題,并不能完全解決問題。shared_ptr可能在將來占主流,它最可能號令江湖, 否則一大堆auto_ptr,weak_ptr,原生指針,scoped_ptr共存就把人搞糊涂了。RFID管理系統集成商 RFID中間件 條碼系統中間層 物聯網軟件集成
    最近免费观看高清韩国日本大全