• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            C++ Programmer's Cookbook

            {C++ 基礎(chǔ)} {C++ 高級(jí)} {C#界面,C++核心算法} {設(shè)計(jì)模式} {C#基礎(chǔ)}

            引領(lǐng)Boost(四)(Boost::smart_ptr)

            Boost::smart_ptr


            一 Boost::smart_Ptr
               
                我們學(xué)習(xí)C++都知道智能指針,例如STL中的std::auto_ptr,但是為什么要使用智能指針,使用它能帶給我們什么好處呢?
                最簡(jiǎn)單的使用智能指針可以不會(huì)因?yàn)橥沝elete指針而造成內(nèi)存泄露。還有如果我們開(kāi)發(fā)或者使用第三方的lib中的某些函數(shù)需要返回指針,這樣的返回的指針被client使用的時(shí)候,lib就會(huì)失去對(duì)返回的指針的控制,這樣delete的指針的任務(wù)一般就會(huì)交給調(diào)用方client,但是如果client忘記調(diào)用delete或是調(diào)用的時(shí)機(jī)不正確,都有可能導(dǎo)致問(wèn)題,在這種情況下就最好使用智能指針。還有使用智能指針可以保證異常安全,保證程序在有異常拋出時(shí)仍然無(wú)內(nèi)存泄露。
               
                std::auto_ptr很多的時(shí)候并不能滿足我們的要求,比如她不能用在STL的container中。boost的smart_ptr中提供了4種智能指針和2種智能指針數(shù)組來(lái)作為std::auto_ptr的補(bǔ)充。   
                  

                shared_ptr<boost/shared_ptr.hpp>:使用shared_ptr進(jìn)行對(duì)象的生存期自動(dòng)管理,使得分享資源所有權(quán)變得有效且安全. 

                scoped_ptr<boost/scoped_ptr.hpp>: 用于確保能夠正確地刪除動(dòng)態(tài)分配的對(duì)象。scoped_ptr 有著與std::auto_ptr類似的特性,而最大的區(qū)別在于它不能轉(zhuǎn)讓所有權(quán)而auto_ptr可以。事實(shí)上,scoped_ptr永遠(yuǎn)不能被復(fù)制或被賦值!scoped_ptr 擁有它所指向的資源的所有權(quán),并永遠(yuǎn)不會(huì)放棄這個(gè)所有權(quán)。 

                weak_ptr<boost/weak_ptr.hpp>:weak_ptr 是 shared_ptr 的觀察員。它不會(huì)干擾shared_ptr所共享的所有權(quán)。當(dāng)一個(gè)被weak_ptr所觀察的 shared_ptr 要釋放它的資源時(shí),它會(huì)把相關(guān)的 weak_ptr的指針設(shè)為空。使用此輔助指針一般是防止懸空指針。

                intrusive_ptr<boost/intrusive_ptr.hpp>:shared_ptr的插入是版本,一般不使用,因?yàn)樾枰獙?duì)使用此指針的類型中增加ref計(jì)數(shù)功能。但是可以保證不增加指針的大小。

                scoped_array<boost/scoped_array.hpp>: scoped_array 為數(shù)組做了scoped_ptr為單個(gè)對(duì)象的指針?biāo)龅氖虑椋核?fù)責(zé)釋放內(nèi)存。
                shared_array<boost/shared_array.hpp>: shared_array 用于共享數(shù)組所有權(quán)的智能指針。一般指向std::vector的shared_ptr提供了比shared_array更多的靈活性,所以一般使用std::vector<shared_ptr>。


            二 源碼剖析

                通過(guò)上面的分析,下面主要介紹我們最常用也最有用的shared_ptr智能指針。(智能指針的實(shí)現(xiàn),其實(shí)最重要的是就是重載->和*運(yùn)算符)

            下面是shared_ptr的頭文件:

            template<class Ty> class shared_ptr {
            public:
              typedef Ty element_type;

              shared_ptr();
              template
            <class Other>
                
            explicit shared_ptr(Other *ptr);
              template
            <class Other, class D>
                shared_ptr(Other 
            *ptr, D dtor);
              shared_ptr(
            const shared_ptr& sp);
              template
            <class Other>
                shared_ptr(
            const shared_ptr<Other>& sp);
              template 
            <class Other>
                shared_ptr(
            const weak_ptr<Other>& wp);
              template
            <class Other>
                shared_ptr(
            const std::auto_ptr<Other>& ap);
              
            ~shared_ptr();

              shared_ptr
            & operator=(const shared_ptr& sp);
              template
            <class Other>
                shared_ptr
            & operator=(const shared_ptr<Other>& sp);
              template
            <class Other>
                shared_ptr
            & operator=(auto_ptr<Other>& ap);

              
            void swap(shared_ptr& s);
              
            void reset();
              template
            <class Other>
                
            void reset(Other *ptr);
              template
            <class Other, class D>
                
            void reset(Other *ptr, D dtor);

              Ty 
            *get() const;
              Ty
            & operator*() const;
              Ty 
            *operator->() const;
              
            long use_count() const;
              
            bool unique() const;
              
            operator boolean-type() const;
              }
            ;


             1)構(gòu)造函數(shù),可以通過(guò)一般指針,std::auto_ptr,boost::shared_ptr,boost::weak_ptr來(lái)構(gòu)造,還可以構(gòu)造的時(shí)候指定如果delete指針。
             2)拷貝構(gòu)造函數(shù)
             3)get(), 得到boost::shared_ptr所封裝的指針。
             4)*,得到boost::shared_ptr所封裝指針的值。
             5)->,用于直接調(diào)用指針的成員。
             6)reset(), 用于重設(shè)boost::shared_ptr,或設(shè)為空。
             7) swap(),  用于交換2個(gè)boost::shared_ptr.
             8) use_count(), use_count 函數(shù)返回指針的引用計(jì)數(shù)。
             9) unique(),這個(gè)函數(shù)在shared_ptr是它所保存指針的唯一擁有者時(shí)返回 true ;否則返回 false。 unique 不會(huì)拋出異常。

             

            三 實(shí)例

             1)構(gòu)造,拷貝構(gòu)造,賦值,get,*,->, reset, swap:

            #include "boost/shared_ptr.hpp"
            #include 
            <cassert>
            #include 
            <iostream>

            class A 
            {  
                boost::shared_ptr
            <int> no_;
            public
                A(boost::shared_ptr
            <int> no) : no_(no) {} 
                
            void value(int i) *no_=i;  }
            }
            ;
            class B 
            {  
                boost::shared_ptr
            <int> no_;
            public:  
                B(boost::shared_ptr
            <int> no) : no_(no) {}  
                
            int value() const {  return *no_;  }
            }
            ;
            struct deleter
            {
                
            void operator()(int *i)
                
            {
                    std::cout 
            << "destroying resource at"
                        
            << (void*)i << '\n';
                    delete i;
                }

            }
            ;
            struct S
            {
                
            int member;
            }
            ;

            int main() 
            {  
                
            // test for constructor
                boost::shared_ptr<int> sp;   
                boost::shared_ptr
            <int> sp2((int*)0);
                
            {
                boost::shared_ptr
            <int> sp3(new int(3), deleter());
                }


                std::auto_ptr
            <int> temp(new int(10));
                boost::shared_ptr
            <int> sp4(temp);

                boost::shared_ptr
            <int> temp2(new int(14));  
                boost::shared_ptr
            <int> sp5(temp2);


                
            // test for using the shared_ptr(get,->,*)
                int *ip = new int(3);                   
                std::cout 
            << (void*)ip << '\n';             
                boost::shared_ptr
            <int> sp6(ip);                
                std::cout 
            << (void*)sp6.get () << '\n';       

                
            int *ip2 = new int(3);               
                std::cout 
            << (void*)ip2 << '\n';         
                boost::shared_ptr
            <int> sp7(ip2);           
                std::cout 
            << *sp7 << '\n';              
                std::cout 
            << (void*)&*sp7 << '\n';        

                S 
            *= new S;                           
                s
            ->member = 4;                           
                boost::shared_ptr
            <S> sp8(s);                    
                std::cout 
            << sp8 -> member << '\n';  

                
            // test for assign 
                std::cout << std::boolalpha;
                boost::shared_ptr
            <int> sp9(new int(0));  
                boost::shared_ptr
            <int> sp10 = sp9;
                std::cout 
            << "sp0 == sp1:" << (sp9 == sp10) << '\n';  
                std::cout 
            << "sp0 != sp2:" << (sp9 != sp10) << '\n';

                
            // test funcion: reset and swap
                boost::shared_ptr<int> sp11 (new int(0));
                boost::shared_ptr
            <int> sp12 (new int(1));
                sp11. swap (sp12);
                std::cout
            <<*sp11<<*sp12<<std::endl;
                boost::swap(sp11, sp12);
                std::cout
            <<*sp11<<*sp12<<std::endl;

                boost::shared_ptr
            <int> sp0; 
                sp0.reset(); 
                boost::shared_ptr
            <int> sp01(new int(1));
                std::cout 
            << *sp01 <<std::endl;
                sp01.reset();
                
            //std::cout << *sp01 <<std::endl; //error
                sp01.reset(new int(100));
                std::cout 
            << *sp01 <<std::endl;
                


                
            // test for query shared_ptr's state (use_count,unique)
                typedef boost::shared_ptr<int> spi;
                spi sp13 ;                       
                std::cout 
            << "empty object: " << sp13.use_count() << '\n';
                spi sp14 ((
            int *)0);               
                std::cout 
            << "null pointer: " << sp14.use_count() << '\n';
                spi sp15 (
            new int);               
                std::cout 
            << "one object: " << sp15.use_count() << '\n';
                
            {
                    spi sp16(sp15);                    
                    std::cout 
            << "two objects: " << sp15.use_count() << '\n';
                    std::cout 
            << "two objects: " << sp16.use_count() << '\n';
                }

                std::cout 
            << "one object: " << sp15.use_count() << '\n';
                
                std::cout 
            << std::boolalpha;
                spi sp17;                       
                std::cout 
            << "empty object: " << sp17.unique() << '\n';
                spi sp18((
            int *)0);              
                std::cout 
            << "null pointer: " << sp18.unique() << '\n';
                spi sp19(
            new int);              
                std::cout 
            << "one object: " << sp19.unique() << '\n';
                

                    spi sp20(sp19);                     
                    std::cout 
            << "two objects: " << sp19.unique() << '\n';
                    std::cout 
            << "two objects: " << sp20.unique() << '\n';
                }
             
                std::cout 
            << "one object: " << sp19.unique() << '\n';

            }

             

             2) 在STL容器中的使用: 

            #include "boost/shared_ptr.hpp"
            #include 
            <vector>
            #include 
            <iostream>
            class A 
            {
            public:  
                
            virtual void sing()
                
            {
                    std::cout 
            <<"A::sing" <<std::endl;
                }

            protected:  
                
            virtual ~A() 
                
            {std::cout<<"~A"<<std::endl;};
            }
            ;
            class B : public A 
            {
            public:  
                
            virtual void sing() 
                
            {   
                    std::cout 
            << "B:sing"<<std::endl; 
                }

                
            virtual ~B()
                
            {std::cout<<"~B"<<std::endl;}
            }
            ;
            boost::shared_ptr
            <A> createA()

                boost::shared_ptr
            <A> p(new B()); 
                
            return p;
            }

            int main() 
            {  
                typedef std::vector
            <boost::shared_ptr<A> > container_type; 
                typedef container_type::iterator iterator; 
                container_type container;
                
            for (int i=0;i<10;++i) 
                
            {   
                    container.push_back(createA()); 
                }
              
                std::cout 
            << "The choir is gathered: \n"
                iterator end
            =container.end(); 
                
            for (iterator it=container.begin();it!=end;++it)
                
            {    (*it)->sing(); }
            }

             

            四 注意

             

            五 參考

            1)Beyond the C++ Standard Library: An Introduction to Boost
            2)boost在線document

             

            posted on 2007-08-22 17:27 夢(mèng)在天涯 閱讀(4580) 評(píng)論(0)  編輯 收藏 引用 所屬分類: CPlusPlus

            公告

            EMail:itech001#126.com

            導(dǎo)航

            統(tǒng)計(jì)

            • 隨筆 - 461
            • 文章 - 4
            • 評(píng)論 - 746
            • 引用 - 0

            常用鏈接

            隨筆分類

            隨筆檔案

            收藏夾

            Blogs

            c#(csharp)

            C++(cpp)

            Enlish

            Forums(bbs)

            My self

            Often go

            Useful Webs

            Xml/Uml/html

            搜索

            •  

            積分與排名

            • 積分 - 1804303
            • 排名 - 5

            最新評(píng)論

            閱讀排行榜

            久久99精品国产| 久久精品无码一区二区日韩AV| 精品无码人妻久久久久久| 精品国产综合区久久久久久| 久久精品无码一区二区WWW| 无码专区久久综合久中文字幕| 久久一日本道色综合久久| 国产成人精品久久免费动漫| 欧美伊香蕉久久综合类网站| 国产精品伊人久久伊人电影| 欧美亚洲另类久久综合婷婷| 2021国内久久精品| 精品国产VA久久久久久久冰| 国产精品熟女福利久久AV| 久久精品人成免费| 国产精品欧美亚洲韩国日本久久| 综合人妻久久一区二区精品| 久久久久久久综合日本亚洲| 午夜精品久久久久久99热| 久久婷婷五月综合色奶水99啪| 狠狠久久综合伊人不卡| 亚洲天堂久久精品| 久久天天躁狠狠躁夜夜不卡 | 久久久久久国产精品美女| 狠狠色伊人久久精品综合网| 久久最近最新中文字幕大全 | 99精品国产免费久久久久久下载 | 国产精品久久成人影院| 亚洲欧美日韩中文久久| 久久99热这里只有精品国产| 久久精品成人欧美大片| 久久久久亚洲精品日久生情| 尹人香蕉久久99天天拍| 久久精品草草草| 韩国三级大全久久网站| 久久久精品一区二区三区| 久久午夜电影网| 国内精品伊人久久久久网站| 久久国产高清一区二区三区| 久久久久久A亚洲欧洲AV冫| 久久亚洲欧洲国产综合|