• <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++書

            游歷C++世界的地圖

            C++小品:weak_ptr與傳達室的花名冊

            原文來自

            http://imcc.blogbus.com/logs/162618478.html

            在C++11中,我們可以使用shared_ptr管理某個對象的所有權,負責對象的析構。然而在某些情況下,我們只是希望安全的訪問某個對象,而不想擁有這個對象的所有權,對這個的析構負責(有點像電視劇中的那些不負責任的男人哦,只是玩玩而已,不會負責)。在這種情況下,我們可以使用表示弱引用的weak_ptr。

            weak_ptr可以由一個shared_ptr構建,表示這個weak_ptr擁有這個shared_ptr所指向的對象的訪問權,注意,這里僅僅是訪問權,它不會改變智能指針的引用計數,自然也就不會去析構這個對象。利用weak_ptr,我們就可以安全地訪問那些不具備所有權的對象。

            一個現實中的例子就是學校的傳達室,傳達室擁有一本學生的名單,如果某個電話來了找某個學生,傳達室會根據花名冊去嘗試訪問這個學生,如果這個學生還在學校,就直接呼叫這個學生,如果已經離開了,這給這個學生留一個消息。在這里,花名冊上的學生可能還在學校(對象還存在),也可能已經離開學校(對象已經析構),我們都需要對其進行訪問,而weak_ptr就是用來訪問這種不確定是否存在的對象的。

            #include <iostream>
            #include <map>
            #include <algorithm>
            #include <memory>
            using namespace std;

            // 學校的同學
            class human
            {
            public:
            human(string _n):name(_n)
            {};
            ~human()
            {
            cout<<name<<" was destructed."<<endl;
            }
            void call() // 嘿,有個電話找你
            {
            cout<<name<<" was called."<<endl;
            }
            private:
            string name;
            };
            // 傳達室
            class doorman
            {
            public:
            doorman(map<string,shared_ptr<human>> humans)
            {
            // 根據學生對象構造花名冊,注意其中保存的是有shared_ptr構造的weak_ptr
            for_each(humans.begin(),humans.end(),
            [&names](pair<string,shared_ptr<human>> h)
            {
            names[h.first] = weak_ptr<human>(h.second);
            });
            }
            // 有個電話打到了傳達室
            void call(string name)
            {
            // 找找看,花名冊中有沒有這個學生
            auto it = names.find(name);
            // 如果有
            if(it!=names.end())
            {
            auto man = (*it).second;
            // 用lock()函數嘗試獲得weak_ptr所指向的shared_ptr,保存為p
            if(auto p = man.lock())
            p->call(); // 如果找到關聯的shared_ptr,也就是這個對象還存在,也就是這個學生還在學校,呼叫之
            else // 如果無法得到關聯的shared_ptr,表示這個對象已經不存在了,學生離開了學校,只能給他留一個消息了
            {
            leavemsg(name);
            }
            }
            else // 如果花名冊中根本沒有這個名字
            {
            cout<<name<<" is not in the school."<<endl;
            }
            }
            void leavemsg(string name)
            {
            cout<<name<<" has left school.I will leave a message for him."<<endl;
            }
            private:
            map<string,weak_ptr<human>> names; // 傳達室的花名冊
            };
            int main()
            {
            // 學校的學生
            map<string,shared_ptr<human>> humans;
            humans["Jiawei"] = make_shared<human>("Jiawei");
            humans["Chen"] = make_shared<human>("Chen");
            humans["Xibei"] = make_shared<human>("Xibei");

            // 傳達室,根據學生構造一個花名冊
            doorman dm(humans);
            // 有人找Chen
            dm.call("Chen");
            // 有人找Fu
            dm.call("Fu");

            // Chen離開學校,對象被析構
            humans.erase("Chen");
            // 又有人打來電話找Chen,這時他已經不在學校,只能給他留一個消息了
            dm.call("Chen");
            // 有人找Jiawei,她還在學校呢,直接叫她
            dm.call("Jiawei");

            return 0;
            }
            從這段程序的輸出,我們也可以看出,我們在刪除humans容器中的Chen這個元素是,對應的human對象也被析構,doorman中指向這個對象的weak_ptr并不影響它的析構,當我們再次嘗試訪問這個對象時候,lock()無法成功獲得與之關聯的shared_ptr,也就無法對其進行訪問了。

            Chen was called.
            Fu is not in the school.
            Chen was destructed.
            Chen has left school.I will leave a message for him.
            Jiawei was called.
            Xibei was destructed.
            Jiawei was destructed.

            這里大家可能會問,為什么不在doorman中使用裸指針呢?。。。

            那么,為什么不直接使用shared_ptr呢? 參考原文。

            總結起來,weak_ptr用于訪問了那些不具備所有權的,可能存在也可能不存在的對象。

            posted on 2011-09-20 16:29 陳良喬——《我的第一本C++書》 閱讀(1812) 評論(0)  編輯 收藏 引用

            国产精品熟女福利久久AV| 99久久99久久| 午夜人妻久久久久久久久| 久久99精品久久久久婷婷| 国产 亚洲 欧美 另类 久久| 国内高清久久久久久| 国产精品xxxx国产喷水亚洲国产精品无码久久一区 | 久久91这里精品国产2020| 日韩久久无码免费毛片软件| 久久中文骚妇内射| 亚洲精品无码久久久| 久久精品无码一区二区三区| 久久无码国产专区精品| 91精品日韩人妻无码久久不卡| 久久久久久一区国产精品| 久久99久久99精品免视看动漫| 亚洲欧美另类日本久久国产真实乱对白 | 一本久久综合亚洲鲁鲁五月天| 日本亚洲色大成网站WWW久久| 国产精品久久久久久福利漫画| 97久久国产综合精品女不卡| 国产高潮久久免费观看| 国产精品99久久久久久人| 人妻精品久久久久中文字幕69 | 国产免费久久久久久无码| 久久超碰97人人做人人爱| 亚洲精品无码久久久久久| 亚洲另类欧美综合久久图片区| 国内精品久久久久影院网站| 久久精品国产秦先生| 国产成人综合久久综合| 久久久久久狠狠丁香| 久久精品成人免费网站| 好久久免费视频高清| 狠狠色丁香婷婷久久综合不卡| 久久综合给合久久国产免费| 久久不见久久见免费视频7| 久久免费视频网站| 国内精品久久久久久不卡影院 | 久久人人爽人人爽人人片AV麻豆| 996久久国产精品线观看|