亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

C++箴言:多態基類中將析構函數聲明為虛擬

2019-11-17 05:12:07
字體:
來源:轉載
供稿:網友

  有很多方法可以跟蹤時間的軌跡,所以有必要建立一個 TimeKeeper 基類,并為不同的計時方法建立派生類:

class TimeKeeper {
 public:
  TimeKeeper();
  ~TimeKeeper();
 ...
};

class AtomicClock: public TimeKeeper { ... };
class WaterClock: public TimeKeeper { ... };
class WristWatch: public TimeKeeper { ... };
  很多客戶只是想簡單地取得時間而不關心如何計算的細節,所以一個 factory 函數——返回一個指向新建派生類對象的基類指針的函數——被用來返回一個指向計時對象的指針:

TimeKeeper* getTimeKeeper(); // returns a pointer to a dynamic-
// ally allocated object of a class
// derived from TimeKeeper
  按照 factory 函數的慣例,getTimeKeeper 返回的對象是建立在堆上的,所以為了避免泄漏內存和其他資源,最重要的就是要讓每一個返回的對象都可以被完全刪除。

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

... // use it

delete ptk; // release it to avoid resource leak
  現在我們精力集中于上面的代碼中一個更基本的缺陷:即使客戶做對了每一件事,也無法預知程序將如何運轉。

  問題在于 getTimeKeeper 返回一個指向派生類對象的指針(比如 AtomicClock),那個對象通過一個基類指針(也就是一個 TimeKeeper* 指針)被刪除,而且這個基類(TimeKeeper)有一個非虛的析構函數。禍端就在這里,因為 C++ 指出:當一個派生類對象通過使用一個基類指針刪除,而這個基類有一個非虛的析構函數,則結果是未定義的。運行時比較有代表性的后果是對象的派生部分不會被銷毀。假如 getTimeKeeper 返回一個指向 AtomicClock 對象的指針,則對象的 AtomicClock 部分(也就是在 AtomicClock 類中聲明的數據成員)很可能不會被銷毀,AtomicClock 的析構函數也不會運行。然而,基類部分(也就是 TimeKeeper 部分)很可能已被銷毀,這就導致了一個古怪的“部分析構”對象。這是一個泄漏資源,破壞數據結構以及消耗大量調試時間的絕妙方法。 排除這個問題非常簡單:給基類一個虛析構函數。于是,刪除一個派生類對象的時候就有了你所期望的正確行為。將銷毀整個對象,包括全部的派生類部分:

class TimeKeeper {
 public:
  TimeKeeper();
  virtual ~TimeKeeper();
  ...
};

TimeKeeper *ptk = getTimeKeeper();
...
delete ptk; // now behaves correctly
  類似 TimeKeeper 的基類一般都包含除了析構函數以外的其它虛函數,因為虛函數的目的就是答應派生類定制實現(參見 Item 34)。例如,TimeKeeper 可能有一個虛函數 getCurrentTime,在各種不同的派生類中有不同的實現。幾乎所有擁有虛函數的類差不多都應該有虛析構函數。

  假如一個類不包含虛函數,這經常預示不打算將它作為基類使用。當一個類不打算作為基類時,將析構函數聲明為虛擬通常是個壞主意??紤]一個表現二維空間中的點的類:

class Point { // a 2D point
 public:
  Point(int xCoord, int yCoord);
  ~Point();
 PRivate:
  int x, y;
};
  假如一個 int 占 32 位,一個 Point 對象正好適用于 64 位的寄存器。而且,這樣一個 Point 對象可以被作為一個 64 位的量傳遞給其它語言寫的函數,比如 C 或者 FORTRAN。假如 Point 的析構函數是虛擬的,情況就完全不一樣了。

  虛函數的實現要求對象攜帶額外的信息,這些信息用于在運行時確定該對象應該調用哪一個虛函數。典型情況下,這一信息具有一種被稱為 vptr(virtual table pointer,虛函數表指針)的指針的形式。vptr 指向一個被稱為 vtbl(virtual table,虛函數表)的函數指針數組,每一個包含虛函數的類都關聯到 vtbl。當一個對象調用了虛函數,實際的被調用函數通過下面的步驟確定:找到對象的 vptr 指向的 vtbl,然后在 vtbl 中尋找合適的函數指針。

  虛函數如何被實現的細節是不重要的。重要的是假如 Point 類包含一個虛函數,這個類型的對象的大小就會增加。在一個 32 位架構中,它們將從 64 位(相當于兩個 int)長到 96 位(兩個 int 加上 vptr);在一個 64 位架構中,他們可能從 64 位長到 128 位,因為在這樣的架構中指針的大小是 64 位的。為 Point 加上 vptr 將會使它的大小增長 50-100%!Point 對象不再適合 64 位寄存器。而且,Point 對象在 C++ 和其他語言(比如 C)中,看起來不再具有相同的結構,因為其它語言缺乏 vptr 的對應物。結果,Points 不再可能傳入其它語言寫成的函數或從其中傳出,除非你為 vptr 做出明確的對應,而這是它自己的實現細節并因此失去可移植性。

  這里的基準就是不加選擇地將所有析構函數聲明為虛擬,和從不把它們聲明為虛擬一樣是錯誤的。實際上,很多人總結過這條規則:當且僅當類中至少包含一個虛擬函數時,則聲明一個虛析構函數。

  但是,當完全沒有虛函數時,就可能和非虛析構函數問題發生撕咬。例如,標準 string 類型不包含虛函數,但是被誤導的程序員有時將它當作基類使用:


class SpecialString: public std::string { // bad idea! std::string has a
... // non-virtual destrUCtor
};
  一眼看上去,這可能無傷大雅,但是,假如在程序的某個地方因為某種原因,你將一個指向 SpecialString 的指針轉型為一個指向 string 的指針,然后你將 delete 施加于這個 string 指針,你就馬上被送入未定義行為的領地。

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.
  同樣的分析可以適用于任何缺少虛析構函數的類,包括全部的 STL 容器類型(例如,vector,list,set,tr1::unordered_map。假如你受到從標準容器類或任何其他帶有非虛析構函數的類派生的誘惑,一定要挺?。。ú恍业氖?,C++ 不提供類似 java 的 final 類或 C# 的 sealed 類的防派生氣制。) 偶然地,給一個類提供一個純虛析構函數能提供一些便利。回想一下,純虛函數導致抽象類——不能被實例化的類(也就是說你不能創建這個類型的對象)。有時候,你有一個類,你希望它是抽象的,但沒有任何純虛函數。怎么辦呢?因為一個抽象類注定要被用作基類,又因為一個基類應該有一個虛析構函數,又因為一個純虛函數產生一個抽象類,好了,解決方案很簡單:在你希望成為抽象類的類中聲明一個純虛析構函數。這是一個例子:

class AWOV { // AWOV = "Abstract w/o Virtuals"
public:
 virtual ~AWOV() = 0; // declare pure virtual destructor
};
  這個類有一個純虛函數,所以它是抽象的,又因為它有一個虛析構函數,所以你不必擔心析構函數問題。這是一個螺旋。你必須為純虛析構函數提供一個定義:

AWOV::~AWOV() {} // definition of pure virtual dtor
  析構函數的工作方式是:最底層的派生類(most derived class)的析構函數最先被調用,然后調用每一個基類的析構函數。編譯器會產生一個從派生類的析構函數對 ~AWOV 的調用,所以你不得不確實為函數提供一個函數體。假如你不這樣做,連接程序會提出抗議。

  為基類提供虛析構函數的規則僅僅適用于多態基類——基類被設計用來答應派生類型通過基類的接口進行操作。TimeKeeper 就是一個多態基類,因為我們期望能操作 AtomicClock 和 WaterClock 對象,甚至當我們僅有指向他們的類型為 TimeKeeper 的指針的時候。

  并非所有的基類都被設計用于多態。例如,無論是標準 string 類型,還是 STL 容器類型都被完全設計成基類,可沒有哪個是多態的。一些類雖然被設計用于基類,但并非被設計用于多態。這樣的類——例如Uncopyable 和標準庫中的 input_iterator_tag——沒有被設計成答應通過基類的接口操作派生類對象。所以它們就不需要虛析構函數。

  Things to Remember

  ·多態基類應該聲明虛析構函數。假如一個類有任何虛函數,它就應該有一個虛析構函數。

  ·假如不是設計用于做基類或不是設計用于多態,這樣的類就不應該聲明虛析構函數。 更多文章 更多內容請看C/C++技術專題專題,或

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产欧美精品一区二区| 国产成人精品免费久久久久| 5566成人精品视频免费| 91麻豆国产精品| 国产精品永久免费在线| 国产一区二区日韩| 庆余年2免费日韩剧观看大牛| 欧美大胆a视频| 亚洲美女av在线播放| 在线日韩精品视频| 国产精品pans私拍| 一夜七次郎国产精品亚洲| 欧美成人免费全部观看天天性色| 亚洲欧美日韩图片| 日本韩国在线不卡| 成人午夜一级二级三级| 欧美一级片在线播放| 国产最新精品视频| 欧洲午夜精品久久久| 国产精品丝袜一区二区三区| 欧美在线视频a| 欧美老女人性生活| 人妖精品videosex性欧美| 亚洲人成电影在线播放| 日韩福利在线播放| 日韩a**站在线观看| 亚洲成人激情在线| 久久精品国产69国产精品亚洲| 色噜噜久久综合伊人一本| 久久久久久久久久av| 久久国产精品偷| 精品亚洲夜色av98在线观看| 成人网欧美在线视频| 日韩免费中文字幕| 日韩大陆欧美高清视频区| 国产日韩中文字幕| xxx欧美精品| 97色在线观看免费视频| 国产在线播放不卡| 中文字幕亚洲专区| 91地址最新发布| 青青草原成人在线视频| 成人免费福利在线| 色悠悠久久久久| 国产一区二区三区日韩欧美| 日韩一区二区久久久| 亚洲最大的免费| 成人在线观看视频网站| 在线丨暗呦小u女国产精品| 日韩精品有码在线观看| 日韩av影院在线观看| 欧美一级电影免费在线观看| 日韩精品在线视频美女| 久久九九精品99国产精品| 清纯唯美亚洲综合| 中文字幕国产亚洲| 欧美老妇交乱视频| 日韩电影免费观看在线| 中文字幕日韩欧美精品在线观看| 久久视频这里只有精品| 在线精品国产欧美| 欧美日韩一区二区三区| 亚洲视频在线免费看| 久色乳综合思思在线视频| 国产精品久久久久久久久久小说| 国产原创欧美精品| 国产精品亚发布| 久久久999精品免费| 日本中文字幕久久看| 久久99久国产精品黄毛片入口| 色婷婷av一区二区三区在线观看| 91精品国产乱码久久久久久久久| 国产成人久久久精品一区| 国产一区二区三区在线免费观看| 欧美激情精品久久久久久免费印度| 国内精品视频久久| 亚洲乱码一区二区| 成人精品视频久久久久| 欧美国产精品va在线观看| 日韩免费不卡av| 91精品久久久久久久久不口人| 国产成人精品av在线| 不卡av电影在线观看| 日韩在线欧美在线国产在线| 在线看片第一页欧美| 欧美性xxxxxxxxx| 国产精品人成电影| 亚洲欧美福利视频| 国产日韩欧美电影在线观看| 欧美色另类天堂2015| 亚洲91精品在线| 国产精品久久久久久久av大片| 91视频九色网站| 亚洲午夜女主播在线直播| 国模叶桐国产精品一区| 欧美一级大片在线免费观看| 亚洲自拍偷拍在线| 国产视频综合在线| 亚洲欧美制服中文字幕| 欧美一区第一页| 欧美大片免费观看在线观看网站推荐| 中文字幕综合一区| 伊人精品在线观看| 欧美激情一区二区三区在线视频观看| 日韩欧美在线国产| 亚洲精品电影在线| 亚洲人午夜精品| 2019亚洲日韩新视频| 免费av在线一区| 国产精品视频网| 国产一区二区三区在线播放免费观看| 国产精品视频男人的天堂| 国产午夜精品免费一区二区三区| 亚洲最新av在线| 国产亚洲精品久久久久动| 亚洲一区二区三| 欧美日韩一区二区三区在线免费观看| 成人免费网站在线看| 国产精品av免费在线观看| 91久久久久久久一区二区| 日韩精品极品视频免费观看| 一区二区三区视频免费| 在线不卡国产精品| 青青久久av北条麻妃黑人| 亚洲激情第一页| 亚洲天堂成人在线| 成人激情视频在线播放| 8050国产精品久久久久久| 国产精品久久久久av免费| 亚洲福利在线播放| 欧美中文字幕精品| 国产精品电影观看| 久久精品欧美视频| 精品丝袜一区二区三区| 久久成年人视频| 国产最新精品视频| 欧美日韩亚洲一区二区| 国产精品美腿一区在线看| 欧美最猛性xxxxx(亚洲精品)| 成人女保姆的销魂服务| 亚洲欧洲一区二区三区久久| 亚洲精品福利在线| 成人在线视频福利| 成人黄色在线播放| 91探花福利精品国产自产在线| 午夜欧美大片免费观看| 国产精品综合网站| 国内精品中文字幕| 成人免费观看a| 亚洲一区二区三区sesese| 欧美一区二区三区精品电影| 最近的2019中文字幕免费一页| 中文字幕日韩在线播放| 日产日韩在线亚洲欧美| 日韩va亚洲va欧洲va国产| 日韩电影视频免费| 大桥未久av一区二区三区| 国产拍精品一二三| 一本一本久久a久久精品综合小说| 成人妇女淫片aaaa视频| 国产一区深夜福利| 国产日韩视频在线观看| 欧美激情一区二区三级高清视频| 秋霞av国产精品一区|