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

牧光小院

被約束的日日夜夜,停不下來的時間。

只在多態(tài)基類中聲明虛析構函數(shù)

關于virtual desctructor的詳細討論。同樣來自于《Effective C++》3rd Edition。

跟蹤時間是很平常的任務,所以開發(fā)一個名為 TimeKeeper 的基類,并讓不同的派生類來實現(xiàn)不同的計時方法是很合理的事情:

class TimeKeeper {

public :

??? TimeKeeper();

??? ~TimeKeeper();

??? ...

};

?

class AtomicClock: public TimeKeeper { ... };

class WaterClock: public TimeKeeper { ... };

class WristWatch: public TimeKeeper{ ... };

很多用戶都希望直接用這些類來計數(shù),而對于他們究竟是如何實現(xiàn)的并不關心。于是一個我們可以用一個 Factory function ——創(chuàng)建一個派生類對象并返回一個基類指針的函數(shù)——返回一個指向 TimeKeeper 的指針。

TimeKeeper* getTimeKeeper(); // returns a pointer to a dynamically

???????????????????????? ? ??? // allocated object of a class derived

???????????????????????? ? ??? // from TimeKeeper

通常, factory function 返回的對象都是創(chuàng)建在堆上的,當用戶使用完計數(shù)器的時候把對象析構掉是很重要的:

TimeKeeper *ptk = getTimeKeeper(); // get dynamically allocated object

??????????????????????????? ?? ????? // from TimeKeeper hierarchy

...???????????????????????? ?? ?????// use it

delete ptk;???????????????????? ?? // release it to avoid resource leak

但是,依賴用戶來執(zhí)行刪除是錯誤的重要來源。條款 18 介紹了如何修改 Factory function 的接口來避免這些常見的用戶錯誤,但是,這些目前都是次要的,因為在上面的代碼中還存在更為嚴重的問題:即使客戶執(zhí)行的正確的動作,你還是無法預期你的程序能夠正確執(zhí)行。

問題在于 getTimeKeeper 返回了一個派生類對象(例如 :AutoicClock ),但是這個對象卻通過基類的指針來刪除(一個指向 Timekeeper 的指針),并且這個基類沒有虛析構函數(shù)。這種組合是制造災難的良方,因為 C++ 規(guī)定:用不帶有虛析構函數(shù)的基類的指針來刪除一個派生類,其結果是未定的。通常在運行時發(fā)生的情況是這個對象的派生類部分沒有被析構。如果 getTimeKeeper 返回一個指向 AtomicClock 對象的指針,那么 AtomicClock 中派生類的部分(例如在 AtomicClock 中聲明的數(shù)據(jù)成員)將不會被正確的析構,實際上 AtomicClock 的析構函數(shù)都根本不會被調(diào)用。但是,基類的部分,卻會被正確的清除,這就造就了一個“畸形”的 partially destroyed object 。這是一個非常棒的泄漏資源、破壞數(shù)據(jù)的方法,它會讓你在調(diào)試器上花費大量的精力。

解決這個問題的方法很簡單,給派生類加上一個虛析構函數(shù)。這樣派生類對象就會如你所愿,被正確的清除:

class TimeKeeper {

public :

??? TimeKeeper();

??? virtual ~TimeKeeper();

??? ...

};

?

TimeKeeper *ptk = getTimeKeeper();

...???????????????????????????
delete ptk;???????????????? // now behaves correctlhy

TimeKeeper 這樣的基類,除了析構函數(shù)外,通常會包含其它的虛函數(shù)。因為虛函數(shù)的目標就是讓派生類來訂制基類的實現(xiàn)。例如, getCurrentTime ,在不同的派生類中就會有不同的實現(xiàn)(注:其實 getTimeKeeper 也可以是一個虛函數(shù))。任何一個擁有虛函數(shù)的類都應該包含一個虛析構函數(shù)。

如果一個類沒有虛函數(shù)呢,這也就意味著這個類并不是被當作基類來使用的。當遇到這種情況的時候,聲明一個虛析構函數(shù)往往不是一個好主意。考慮一個用來表示二維空間中的某點的類:

class Point {// a 2D point

public :

??? Point(int xCoord, int yCoord);

??? ~Point();

?

private :

??? int x, y;
};

如果一個 int 32 bits ,這樣的一個 Point 可以被放到一個 64 位寄存器中。另外,這樣的一個 Point 對象還可以被當作是一個整體被其它的語言使用,例如 C FORTRAN 。但是,如果 Point 的析構函數(shù)是虛擬的,故事就完全不一樣了。

虛函數(shù)的實現(xiàn)需要對象承載某些額外信息,這些信息用來在運行時對虛函數(shù)的調(diào)用進行正確的轉(zhuǎn)發(fā)。這個額外的信息使通過一個 vtpr 來實現(xiàn)的。 Vptr 指向一個存放函數(shù)指針( vtbl )的數(shù)組,每一個具有虛函數(shù)的類都有一個對應的 vtbl 。當一個對象的虛函數(shù)被調(diào)用的時候,該對象的 vtpr vtbl 組合來完成定位正確的函數(shù)調(diào)用的工作。

這里,虛函數(shù)如何實現(xiàn)的并不重要。重要的是如果 Point 包含了一個虛函數(shù),對象將會長胖。在一個 32 bits 的機器上,它將會從 64 bits 長到 96 bits ;在 64 bit 的機器上,它將會從 64 bits 長到 128 bits 。這個額外的 vtpr 的存在讓對象的體積增長了 50%~100% Point 對象也不再能夠放到一個 64-bit 寄存器中了。另外, Point 對象也不再和 C 語言的保持兼容,因為 C 語言中沒有 vrpr 機制。結果是,你要想使用該 Point 對象,除非自己來實現(xiàn) vtpr vtpl 機制,而這樣做,往往又會降低你的代碼的可移植性。

也就是說,把所有的析構函數(shù)都不加思索的聲明為虛擬的和從不把它們聲明為虛擬的一樣,都是不明智的行為。實際上,很多人得除了這樣的結論:當且僅當一個類有至少一個虛函數(shù)的時候,才把析構函數(shù)聲明為虛擬的。

實際上,即使你的類中沒有虛函數(shù),你還是有可能被非虛析構函數(shù)的問題咬上一口。例如 std::string 就沒有虛函數(shù),但是一些被誤導的程序員有時會把它當作基類來使用:

class SpecialString: public std::string {

// bad idea! std::string has a

??? ...????????????????????? ??? // non-virtual destructor
}

乍一看,這可能沒什么問題,但是一旦你把一個指向 SpecialString 的指針轉(zhuǎn)換成一個 string ,并用這個指針來刪除 SpecialString 對象的時候,你馬上就被帶進了未定義行為的深潭。

SpecialString *pss = new SpecialString("Impending Doom");

std::string *ps;

...

ps = pss;? // SpecialString* --> std::string*

?

delete ps;? // undefined! In practice, *ps's Specialstring resources

?????????? ? // will be leaked, because the SpecialString destructor won't??????? // be called

同樣的結果還會出現(xiàn)在其它沒有虛析構函數(shù)的類中,例如所有的 STL 容器類型(例如: vector, list, set, tr1::unordered_map 等等)。如果你曾經(jīng)對于從一個標準容器或其它帶有非虛析構函數(shù)的類繼承,那么徹底打消這個想法。(不幸的是 C++ 沒有提供像 C#(sealed) Java(final) 類似的拒絕繼承的語言機制)

有時候,把析構函數(shù)設定為 pure virtual 是非常方便的。一個 pure virtual 函數(shù)可以讓一個類成為抽象類。有時,你可能需要讓你的類成為一個 abstract class ,但是你一時又找不到合適的純虛函數(shù)。怎么辦呢?因為一個抽象類往往是要被作為基類的,而一個基類往往又應該有一個虛析構函數(shù)。這樣一來:聲明一個 pure virtual destructor 就是一個不錯的主意。一箭雙雕。

class AWOV {? // AWOV = "Abstract w/o Virtuals"

public :

??? virtual ~AWOV() = 0; // declare pure virtual destructor

};

這個類有一個純虛函數(shù),因此這是以個抽象基類,并且這個類有一個虛析構函數(shù),這也使你遠離了析構函數(shù)的問題,唯一要注意的,就是一定要為純虛析構函數(shù)提供一份實現(xiàn)。

虛析構函數(shù)的工作方式是從最深的派生類的析構函數(shù)依次調(diào)用其基類的析構函數(shù),編譯器會生成生成一個從派生類到基類的 ~AWOV 的調(diào)用。如果你沒有提供析構函數(shù)的實現(xiàn),鏈接器就會抱怨錯誤。

所以,你只應該把多態(tài)基類的析構函數(shù)聲明為虛擬的。只有你想通過基類接口來操作派生類的時候,一個基類才是多態(tài)的。 TimeKeeper 就是一個多態(tài)基類,因為我們需要用一個 TimeKeeper* 來操作 AtomicClock WaterClock 對象。

另外,并不是所有的基類都要按照多態(tài)的方式來設計和使用。 Std::string STL 中的容器類型就都不具備多態(tài)性。一些類被設計成基類,但是卻不應該按照多態(tài)的方式來使用,例如 input_iterator_tag 就是一個例子,你并不需要用基類接口來操縱派生類。結果是,他們也不需要虛擬析構函數(shù)。

時時刻刻讓自己記住

l ???????? 應該為多態(tài)基類聲明虛擬析構函數(shù)。如果一個類有一個虛函數(shù),那么它也應該有一個虛析構函數(shù)

l ???????? 如果一個類不是被設計為基類或者它們并不是按照多態(tài)的方式來使用的,不要為它們聲明虛析構函數(shù)

posted on 2005-11-10 16:43 nacci 閱讀(2199) 評論(3)  編輯 收藏 引用 所屬分類: C++漫談

評論

# re: 只在多態(tài)基類中聲明虛析構函數(shù) 2005-11-24 12:18 sdfsd

gfdsg  回復  更多評論   

# re: 只在多態(tài)基類中聲明虛析構函數(shù) 2006-02-28 10:11 zzq

好貼
  回復  更多評論   

# re: 只在多態(tài)基類中聲明虛析構函數(shù) 2006-03-14 14:56 hhxz

這個文章不錯,有意思  回復  更多評論   

<2025年9月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

導航

統(tǒng)計

常用鏈接

留言簿(2)

隨筆分類

收藏夾

大家的聲音

積分與排名

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲国产欧美一区二区三区久久 | 亚洲精选一区二区| 欧美日韩一区视频| 亚洲国产91| 毛片精品免费在线观看| 欧美综合77777色婷婷| 欧美日韩妖精视频| 亚洲精品日韩精品| 欧美久久成人| 免费观看成人鲁鲁鲁鲁鲁视频 | 久久精品视频网| 久久福利影视| 性欧美8khd高清极品| 午夜精品一区二区三区电影天堂| 午夜欧美理论片| 午夜在线精品偷拍| 欧美一级二级三级蜜桃| 久久久久久久一区二区三区| 久久久久久久性| 久久蜜桃精品| 亚洲高清二区| 一本色道久久88精品综合| 艳女tv在线观看国产一区| 久久免费视频这里只有精品| 亚洲国产另类 国产精品国产免费| 欧美高清成人| 亚洲精品中文字幕在线| 欧美在线观看日本一区| 久久综合一区二区| 欧美区日韩区| 在线看日韩欧美| 99天天综合性| 欧美影院一区| 亚洲精品日韩精品| 亚洲尤物在线视频观看| 欧美综合二区| 国产精品夜夜夜一区二区三区尤| 国产主播一区二区三区| 亚洲美女视频网| 久久人人爽人人| 亚洲精品偷拍| 欧美中文字幕第一页| 夜夜狂射影院欧美极品| 亚洲欧美视频在线| 亚洲电影免费观看高清| 国产精品任我爽爆在线播放| 国产精品久久一卡二卡| 欧美日韩亚洲一区二区三区四区| 国产精品久久久久久久久久直播 | 国产精品欧美经典| 免费亚洲视频| 欧美了一区在线观看| 欧美日韩一区二区在线观看视频| 久久精品视频免费| 久久精品91久久香蕉加勒比| 免费看亚洲片| 欧美承认网站| 国产精品美女主播| 亚洲激情av| 久久久www成人免费精品| 亚洲视频在线观看| 欧美成人a视频| 国产亚洲欧美日韩日本| 性欧美18~19sex高清播放| 亚洲精品日产精品乱码不卡| 久久亚洲国产精品一区二区| 久久久久综合网| 亚洲欧美日韩精品久久| 欧美黄色一区二区| 亚洲福利在线观看| 亚洲第一色中文字幕| 欧美在线视频a| 国产精品专区第二| 久久久久**毛片大全| 亚洲视频欧美视频| 欧美人在线观看| 亚洲欧美美女| 亚洲视频在线一区| 国产精品v日韩精品v欧美精品网站| 亚洲一区二区三区色| 亚洲精品久久7777| 99精品国产在热久久下载| 亚洲国产成人一区| 亚洲国产精品va| 久久综合伊人77777蜜臀| 激情另类综合| 亚洲欧美日韩一区二区三区在线| 最新精品在线| 欧美啪啪一区| 亚洲欧美综合v| 亚洲国产精品ⅴa在线观看 | 亚洲专区在线视频| 亚洲无限av看| 国产亚洲欧美一区在线观看 | 欧美日韩国产精品一区二区亚洲| 99re这里只有精品6| 日韩视频一区| 欧美日韩在线三区| 久久久国产视频91| 六月丁香综合| 中文亚洲免费| 久久精品夜色噜噜亚洲a∨| 一区在线视频观看| 亚洲国产精品传媒在线观看| 国产精品免费看| 毛片一区二区| 欧美精品一区二区三区蜜桃| 久久精品视频播放| 欧美成人一区二免费视频软件| 在线视频你懂得一区| 欧美一区二区私人影院日本| 在线日韩av永久免费观看| 亚洲久久成人| 亚洲大胆人体视频| 在线综合亚洲欧美在线视频| 国产日韩三区| 亚洲视频观看| 亚洲第一精品福利| 99热精品在线观看| 亚洲品质自拍| 欧美一区二区视频免费观看| 亚洲美女在线看| 先锋影院在线亚洲| 99精品热6080yy久久| 欧美亚洲综合另类| 午夜精品久久久久久久久| 欧美国产极速在线| 久久久爽爽爽美女图片| 国产精品久久久久久久久久久久久久| 久久国产精品毛片| 欧美色视频在线| 亚洲精品日韩在线观看| 亚洲福利视频三区| 欧美资源在线观看| 久久久青草青青国产亚洲免观| 欧美日韩精品三区| 欧美激情第1页| 国产一本一道久久香蕉| 亚洲精品乱码久久久久久蜜桃91| 欧美国产第一页| 国产日韩在线一区| 老司机精品导航| 国产伊人精品| 亚洲愉拍自拍另类高清精品| 亚洲福利在线看| 久久综合狠狠| 久热国产精品视频| 久久视频在线免费观看| 久久久久久网站| 国产精品永久免费在线| 一个色综合av| 午夜精品在线| 国产精品国产精品国产专区不蜜| 久久黄色级2电影| 国产精品久久一级| 亚洲一区二区三区中文字幕在线 | 香蕉久久国产| 在线观看成人av| 蜜月aⅴ免费一区二区三区| 久久日韩粉嫩一区二区三区| 欧美专区日韩视频| 亚洲一区二区三区精品在线观看 | 一片黄亚洲嫩模| 亚洲国产高清一区二区三区| 亚洲国产精品精华液2区45| 亚洲视频在线一区| 一本久久综合亚洲鲁鲁| 亚洲一区二区伦理| 国产一区二区三区免费不卡| 久久精品成人| 亚洲福利国产| 亚洲系列中文字幕| 国产日韩av一区二区| 久久久www成人免费无遮挡大片| 欧美国产日产韩国视频| 午夜精彩视频在线观看不卡| 黄色精品一区| 欧美日韩成人在线观看| 午夜精品在线看| 欧美激情精品久久久久久久变态| 亚洲看片一区| 国产一区二区三区在线免费观看 | 久久精品最新地址| 亚洲美女免费视频| 久久一区精品| 亚洲一区欧美激情| 亚洲国产一区视频| 国产亚洲欧美一区在线观看 | 亚洲永久免费视频| 亚洲欧美影音先锋| 亚洲欧洲精品一区二区三区不卡 | 91久久精品国产91久久性色| 亚洲午夜激情| 国产精品久久二区| 性伦欧美刺激片在线观看| 欧美成人高清| 在线观看国产精品淫| 亚洲午夜久久久久久久久电影网| 狠狠综合久久| 欧美激情导航|