青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

清風竹林

ぷ雪飄絳梅映殘紅
   ぷ花舞霜飛映蒼松
     ----- Do more,suffer less

Solmyr 的小品文系列之五:垃圾收集

午餐時間。
zero 坐在餐桌前,機械的重復“夾菜 -> 咀嚼 -> 吞咽”的動作序列,臉上用無形的大字寫著:我心不在焉。在他的對面坐著 Solmyr ,慢條斯理的吃著他那份午餐,維持著他一貫很有修養(yǎng)的形象 ——— 或者按照 zero 這些熟悉他本質(zhì)的人的說法:假象。

“怎么了 zero ?胃口不好么?”,基本填飽肚子之后,Solmyr 覺得似乎應該關心一下他的學徒了。

“呃,沒什么,只是 …… Solmyr ,C++ 為什么不支持垃圾收集呢?(注:垃圾收集是一種機制,保證動態(tài)分配了的內(nèi)存塊會自動釋放,Java 等語言支持這一機制。)”

Solmyr 嘆了口氣,用一種平靜的眼神盯著 zero :“是不是在 BBS 上和人吵 C++ 和 Java 哪個更好?而且吵輸了?我早告訴過你,這種爭論再無聊不過了。”

“呃 …… 是”,zero 不得不承認 ——— Solmyr 的眼神雖然一點也不銳利,但是卻莫名其妙的讓 zero 產(chǎn)生了微微的恐懼感。

“而且,誰告訴你 C++ 不支持垃圾收集的?”

“啊!Solmyr 你不是開玩笑吧?!”

“zero 你得轉(zhuǎn)變一下觀念。我問你,C++ 支不支持可以動態(tài)改變大小的數(shù)組?”

“這 …… 好象也沒有吧?”

“那 vector 是什么東西?”

“呃 ……”

“支持一種特性,并不是說非得把這個特性加到語法里去,我們也可以選擇用現(xiàn)有的語言機制實現(xiàn)一個庫來支持這個特征。以垃圾收集為例,這里我們的任務是要保證每一個被動態(tài)分配的內(nèi)存塊都能夠被釋放,也就是說 ……”,Solmyr 不知從哪里找出了一張紙、一支筆,寫到:

int* p = new int;        // 1
delete p;                // 2

“也就是說,對于每一個 1 ,我們要保證有一個 2 被調(diào)用,1 和 2 必須成對出現(xiàn)。我來問你,C++ 中有什么東西是由語言本身保證一定成對出現(xiàn)的?”

“……”,zero 露出了努力搜索記憶的表情,不過很明顯一無所獲。

“提示一下,和類的創(chuàng)建有關。”

“哦!構造函數(shù)與析構函數(shù)!”

“正確。可惜普通指針沒有構造函數(shù)與析構函數(shù),所以我們必須要寫一個類來加一層包裝,最簡單的就象這樣:”

class my_intptr
{
public:
    int* m_p;

    my_intptr(int* p){ m_p = p; }
    ~my_intptr(){ delete m_p; }
};

…………

my_intptr pi(new int);
*(pi.m_p) = 10;

…………

“這 里我們可以放心的使用 my_intptr ,不用擔心內(nèi)存泄漏的問題:一旦 pi 這個變量被銷毀,我們知道 pi.p 指向的內(nèi)存塊一定會被釋放。不過如果每次使用 my_intptr 都得去訪問它的成員未免太麻煩了。為此,可以給這個類加上重載的 * 運算符:”

class my_intptr
{
private:
    int* m_p;

public:
    my_intptr(int* p){ m_p = p; }
    ~my_intptr(){ delete m_p; }

    int& operator*(){ return *m_p; }
};

…………

my_intptr pi;
*pi = 10;
int a = *pi;

…………

“現(xiàn)在是不是看起來 my_intptr 就像是一個真正的指針了?正因為如此,這種技術被稱為智能指針。現(xiàn)在我問你,這個類還缺少哪些東西?”

zero 皺著眉頭,眼睛一眨一眨,看上去就像一臺慢速電腦正在辛苦的往它的硬盤上拷貝文件。良久,zero 抬起頭來,不太確定的說:“是不是還缺少一個拷貝構造函數(shù)和一個賦值運算符?”

“說說為什么。”,Solmyr 顯然不打算就這樣放過 zero。

“因為 …… 我記得沒錯的話 …… 《50 誡 》(注: 指《Effective C++ 2/e》一書)中提到過,如果你的類里面有指針指向動態(tài)分配的內(nèi)存,那么一定要為它寫一個拷貝構造函數(shù)和一個賦值運算符 …… 因為 …… 否則的話,一旦你做了賦值,會導致兩個對象的指針指向同一塊內(nèi)存。對了!如果是上面的類,這樣一來會導致同一個指針被 delete 兩次!”

“正確。那么我們應該怎樣來實現(xiàn)呢?”

“這簡單,我們用 memcpy 把目標指針指向的內(nèi)存中的內(nèi)容拷貝過來。”

“如果我們的智能指針指向一個類的對象怎么辦?注意,類的對象中可能有指針,不能用 memcpy。”

“那 …… 我們用拷貝構造的辦法。”

“如果我們的智能指針指向的對象不能拷貝構造怎么辦?它可能有一個私有的拷貝構造函數(shù)。”

“那 ……”,zero 頓了一頓,決定老實承認,“我不知道。”

“問題在哪你知道么?在于你沒有把智能指針看作指針。想象一下,如果我們對一個指針做賦值,它的含義是什么?”

“呃,我明白了,在這種情況下,應該想辦法讓兩個智能指針指向同一個對象 …… 可是 Solmyr ,這樣以來豈不是仍然要對同一個對象刪除兩遍?”

“是 的,我們得想辦法解決這個問題,辦法不只一種。比較好的一種是為每個指針維護一個引用計數(shù)值,每次賦值或者拷貝構造,就讓計數(shù)值加一,這意味著指向這個內(nèi) 存塊的智能指針又多了一個;而每有一個智能指針被銷毀,就讓計數(shù)值減一,這意味著指向這個內(nèi)存塊的智能指針少了一個;一旦計數(shù)值為 0 ,就釋放內(nèi)存塊。象這樣:”

class my_intptr
{
private:
    int* m_p;
    int* m_count;

public:
    my_intptr(int* p)
    {
        m_p = p;
        m_count = new int;             // 初始化計數(shù)值為 1
        *m_count = 1;
    }
    my_intptr(const my_intptr& rhs)    // 拷貝構造函數(shù)
    {
        m_p = rhs.m_p;                 // 指向同一塊內(nèi)存
        m_count = rhs.m_count;         // 使用同一個計數(shù)值
        (*m_count)++;                  // 計數(shù)值加 1
    }
    ~my_intptr()
    {
        (*m_count)--;           // 計數(shù)值減 1
        if( *m_count == 0 )     // 已經(jīng)沒有別的指針指向該內(nèi)存塊了
        {
            delete m_p;
            delete m_count;
        }
    }

    my_intptr& operator=(const my_intptr& rhs)
    {
        if( m_p == rhs.m_p )        // 首先判斷是否本來就指向同一內(nèi)存塊
            return *this;           // 是則直接返回

        (*m_count)--;               // 計數(shù)值減 1 ,因為該指針不再指向原來內(nèi)存塊了
        if( *m_count == 0 )         // 已經(jīng)沒有別的指針指向原來內(nèi)存塊了
        {
            delete m_p;
            delete m_count;
        }

        m_p = rhs.m_p;            // 指向同一塊內(nèi)存
        m_count = rhs.m_count;    // 使用同一個計數(shù)值
        (*m_count)++;             // 計數(shù)值加 1
    }

    …………
};

“其他部分沒有什么太大變化,我不費事了。現(xiàn)在想象一下我們怎樣使用這種智能指針?”,Solmyr 放下了筆,再次拿起了筷子,有些惋惜的發(fā)現(xiàn)他愛吃的肉丸子已經(jīng)冷了。

zero 想象著,有些遲疑。“我們 …… 可以用 new int 表達式作為構造函數(shù)的參數(shù)來構造一個智能指針,然后 …… 然后我們可以任意的賦值,”,他開始抓住了思路,越說越快,“任意的用已經(jīng)存在的智能指針來構造新的智能指針,智能指針的賦值運算符、拷貝構造函數(shù)和析構 會保證計數(shù)值始終等于指向該內(nèi)存塊的智能指針數(shù)。”zero 似乎明白了他看到了怎樣的功能,開始激動起來:“然后一旦計數(shù)值為 0 被分配的內(nèi)存塊就會釋放!也就是說 …… 有指針指向內(nèi)存塊,它就不釋放,一旦沒有,它就自動釋放!太棒了!我們只要一開始正確的初始化智能指針,就可以象普通指針那樣使用它,而且完全不用擔心內(nèi) 存釋放的問題!太棒了!”zero 激動的大叫:“這就是垃圾收集!Solmyr !我們在飯桌上實現(xiàn)了一個垃圾收集器!”

Solmyr 很明顯沒有分享 zero 的激動:“我在吃飯,你能不能不要大叫‘飯桌上實現(xiàn)了一個垃圾收集器’這種倒胃口的話?”頓了一頓,Solmyr 帶著他招牌式的壞笑,以一種可惡的口吻說道:“而且請注意一下自己的形象。”

“嗯?”,zero 回過神來,發(fā)現(xiàn)自己不知什么時候站了起來,而整個餐廳里的人都在看著他嘿嘿偷笑,這讓他感覺自己像個傻瓜。

zero 紅著臉坐下,壓低了聲音問 Solmyr :“不過 Solmyr ,這確實是一個的垃圾收集機制啊,只要我們把這個類改成 …… 嗯 …… 改成模板類,象這樣:”zero 抓過了紙筆,寫到:

template <typename T>
class my_ptr
{
private:
    T* m_p;
    int* m_count;
    …………
};

“它不就能支持任意類型的指針了嗎?我們就可以把它用在任何地方。”

Solmyr 搖了搖頭:“不,你把問題想的太簡單了。對于簡單的類型,這個類確實可以處理的很好,但實際情況是很復雜的。考慮一個典型情況:類 Derived 是類 Base 的派生類,我們希望這樣賦值:”

Base* pb;
Derived pd;
…………
pb = pd;

“你倒說說看,這種情況,怎樣改用上面這個智能指針來處理?”

“……”,zero 沉默了。

“要 實現(xiàn)一個完整的垃圾收集機制并不容易,因為有許多細節(jié)要考慮。”,Solmyr 開始總結了,“不過,基本思路就是上面說的這些。值得慶幸的是,目前已經(jīng)有了一個相當成熟的‘引用計數(shù)’智能指針,boost::shared_ptr。 大多數(shù)情況下,我們都可以使用它。另外,除了智能指針之外,還有一些技術也能夠幫助我們避開釋放內(nèi)存的問題,比如內(nèi)存池。但是,關鍵在于 ——— ”

Solmyr 再度用那種平靜的眼神盯著 zero :

“身為 C/C++ 程序員,必須有創(chuàng)造力。那種躺在語言機制上不思進取的人,那種必須要靠語法強制才知道怎樣編程的人,那種沒有別人告訴他該干什么就無所適從的人,不適合這門語言。”

posted on 2009-08-19 11:05 李現(xiàn)民 閱讀(517) 評論(0)  編輯 收藏 引用 所屬分類: 絕對盜版

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            老**午夜毛片一区二区三区| 日韩一级片网址| 国产一区自拍视频| 欧美黄色成人网| 欧美一区二区三区在| 性久久久久久| 欧美黑人多人双交| 欧美精品xxxxbbbb| 欧美人成网站| 国产亚洲制服色| 在线视频你懂得一区| 一区二区三区国产在线观看| 国产麻豆精品视频| 黄色亚洲精品| 亚洲图色在线| 亚洲第一精品电影| 亚洲区欧美区| 欧美一区二区视频97| 欧美成人精品| 亚洲精品永久免费| 一本色道久久88精品综合| 亚洲毛片av在线| 欧美国产日韩精品免费观看| 欧美日韩国产综合视频在线观看中文| 国产婷婷色一区二区三区在线| 亚洲国产成人久久| 久久久高清一区二区三区| 亚洲国产精品毛片| 久久久久久亚洲精品杨幂换脸| 国产精品video| 在线综合亚洲| 日韩亚洲精品电影| 欧美激情一区二区三区全黄| 伊人成人在线| 亚洲国产免费看| 欧美va天堂在线| 一二三区精品| 亚洲性感激情| 亚洲高清不卡av| 亚洲三级毛片| 国产精品亚洲成人| 久久久成人网| 久久综合亚州| 9色精品在线| 性欧美精品高清| 亚洲第一网站| 亚洲蜜桃精久久久久久久 | 在线观看福利一区| 久久蜜桃香蕉精品一区二区三区| 久久精品中文字幕一区二区三区| 在线视频观看日韩| 99综合视频| 亚洲毛片在线免费观看| 欧美在线啊v一区| 西瓜成人精品人成网站| 亚洲高清在线视频| 在线午夜精品自拍| 在线欧美小视频| 在线亚洲伦理| 日韩视频―中文字幕| 午夜亚洲视频| 亚洲小视频在线| 免费视频亚洲| 欧美大片一区| 亚洲国产cao| 久久久久久久久久久一区| 99在线视频精品| 欧美精品一区视频| 亚洲高清视频在线观看| 禁久久精品乱码| 欧美一区二区播放| 久久综合狠狠综合久久综青草 | 欧美华人在线视频| 久久久美女艺术照精彩视频福利播放| 欧美三级视频| 香蕉成人久久| 久久永久免费| 91久久精品视频| 欧美a级大片| 在线午夜精品| 久久青草福利网站| 一区二区日韩伦理片| 欧美午夜女人视频在线| 亚洲欧美日韩精品久久久久| 久久精品国产免费| 亚洲精品一区二区三区不| 欧美日韩国产一中文字不卡 | 亚洲尤物在线| 国产精品99久久久久久宅男| 欧美日韩在线视频首页| 亚洲一区二区网站| 免费成人高清视频| 亚洲愉拍自拍另类高清精品| 欧美另类一区| 久久久综合香蕉尹人综合网| 亚洲国产欧美一区二区三区久久 | 麻豆国产精品777777在线| 亚洲激情图片小说视频| 国产视频久久久久久久| 欧美va亚洲va国产综合| 亚洲一区中文| 一区二区三区四区五区视频| 亚洲成人中文| 麻豆精品精华液| 久久国产毛片| 午夜精品福利在线| 亚洲欧美精品在线观看| 一区二区精品| 亚洲小说春色综合另类电影| 亚洲国产精品成人一区二区 | 麻豆久久精品| 欧美专区第一页| 久久久久久噜噜噜久久久精品| 性欧美xxxx大乳国产app| 亚洲午夜精品在线| 一区二区三区四区国产精品| 99国产精品私拍| 一本久久a久久免费精品不卡| 亚洲精品国产品国语在线app | 欧美性猛交视频| 国产精品久久久久久久久久久久| 欧美福利视频在线观看| 欧美日韩精品在线视频| 欧美久久久久久| 韩国女主播一区二区三区| 在线观看日韩av| 国产精品99久久久久久久女警| 一本久久综合| 久久精品国产99精品国产亚洲性色 | 久久精品国产亚洲a| 久久资源av| 国产一区二区三区视频在线观看 | 亚洲精品一区二区三区99| 亚洲精品日韩精品| 香蕉成人久久| 欧美多人爱爱视频网站| 国产欧美日韩三区| 亚洲精品欧美专区| 久久国产日韩| 亚洲一级黄色片| 欧美四级电影网站| 999亚洲国产精| 亚洲国产高清高潮精品美女| 久久疯狂做爰流白浆xx| 国产日产欧产精品推荐色| 中文国产一区| 亚洲人成人一区二区在线观看| 日韩亚洲一区二区| 亚洲电影免费观看高清完整版在线观看 | 久久久久国产精品www| 亚洲一区一卡| 国产一区久久久| 蜜月aⅴ免费一区二区三区| 久久激情五月激情| 亚洲黄色在线看| 亚洲人体偷拍| 欧美乱人伦中文字幕在线| 夜夜嗨av一区二区三区四季av | 欧美伊人久久久久久久久影院| 亚洲天堂激情| 国产亚洲福利| 亚洲国产日韩欧美在线99| 欧美天堂亚洲电影院在线播放| 亚洲在线视频免费观看| 亚洲美女毛片| 国产日韩精品入口| 欧美二区在线| 国产精品久久久久久久久果冻传媒 | 欧美精品一区在线| 性一交一乱一区二区洋洋av| 亚洲一区二区三区四区视频| 国产欧美日韩综合一区在线播放| 久久在线视频| 国产日产欧美精品| 亚洲精品一线二线三线无人区| 欧美日韩在线直播| 久久久精品日韩欧美| 欧美在线亚洲| 欧美激情影院| 亚洲黄页视频免费观看| 欧美日韩亚洲一区三区 | 亚洲国产视频一区二区| 亚洲欧美乱综合| 久久亚洲精品一区二区| 亚洲影院色无极综合| 欧美日韩黄色大片| 一本色道久久综合精品竹菊| 亚洲免费观看在线视频| 免费观看成人| 亚洲精品免费在线观看| 亚洲精品视频在线播放| 欧美精品videossex性护士| 亚洲精品女人| 在线亚洲伦理| 国产视频亚洲精品| 午夜精品一区二区三区四区 | 亚洲国产成人久久综合一区| 欧美成人精品一区| 久久精品人人做人人爽电影蜜月| 久久久久一区二区|