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

洛譯小筑

別來無恙,我的老友…
隨筆 - 45, 文章 - 0, 評論 - 172, 引用 - 0
數(shù)據(jù)加載中……

[ECPP讀書筆記 條目34] 區(qū)分清接口繼承和實現(xiàn)繼承

公共繼承的概念看似簡單,似乎很輕易就浮出水面,然而仔細審度之后,我們會發(fā)現(xiàn)公共繼承的概念實際上包含兩個相互獨立的部分:函數(shù)接口的繼承和函數(shù)實現(xiàn)的繼承。二者之間的差別恰與函數(shù)聲明和函數(shù)實現(xiàn)之間相異之處(本書引言中有介紹)等價。

假如你是一個類設(shè)計人員,某些場合下你需要使派生類僅僅繼承基類成員函數(shù)的接口(聲明)。而另一些時候你需要讓派生類繼承將函數(shù)的接口和實現(xiàn)都繼承過來,但還期望可以覆蓋繼承而來的具體實現(xiàn)。另外,你還可能會希望在派生類中繼承函數(shù)的接口和實現(xiàn),而不允許覆蓋任何內(nèi)容。

為了獲取上述三種選項的直觀感受,可以參考下面的類層次結(jié)構(gòu)實例,該實例用于在圖形程序中表示幾何形狀:

class Shape {

public:

  virtual void draw() const = 0;

  virtual void error(const std::string& msg);

  int objectID() const;

 

  ...

};

 

class Rectangle: public Shape { ... };

class Ellipse: public Shape { ... };

Shape是一個抽象類,純虛函數(shù)draw標示著這一點。因此客戶便無法創(chuàng)建Shape類的實例,只能由Shape類繼承出新的派生類。不過,Shape對所有由它(公共)繼承出的類有著較強的影響,因為

成員函數(shù)的接口總會被繼承下來。就像條目32中所解釋的,公共繼承意味著“A是一個B”的關(guān)系,因此對于基類成立的任何東西,對于派生類也應(yīng)成立。由此可知,如果一個函數(shù)對某個類適用,那么它同樣也適用于這個類的派生類。

Shape類中聲名了三個函數(shù),第一個是draw,用于把當前對象繪制在一個假想的顯示設(shè)備上。第二個是error,在成員函數(shù)需要報告錯誤時它將被調(diào)用。第三個是objectID,為當前對象返回一個標識身份的整數(shù)值。每個函數(shù)的聲明方式各不相同:draw是一個純虛函數(shù),error是一個簡單(非純虛的)虛函數(shù),objectID是一個非虛函數(shù)。那么這些不同的聲明方式的具體實現(xiàn)又是什么樣的呢?

首先請看純需函數(shù)draw

class Shape {

public:

  virtual void draw() const = 0;

  ...

};

純虛函數(shù)最為顯著的兩個特征是:首先,在所有派生出的實體類中,必須要對它們進行重新聲明;其次,純虛函數(shù)在抽象類中一般沒有定義內(nèi)容。融合以上兩點我們可以看出:

聲明純虛函數(shù)的目的就是讓派生類僅僅繼承函數(shù)接口

對于Shape::draw函數(shù)來說上面的分析再恰當不過了,因為“所有的Shape對象必須能夠繪制出來”這一要求十分合理,但是“由Shape類來提供缺省的具體實現(xiàn)”就顯得很牽強了。比如說,繪制一個橢圓的算法與繪制一個長方形大相徑庭。Shape::draw告訴具體派生類的設(shè)計者:你必須要提供一個draw函數(shù),但是我可不知道你要怎么去實現(xiàn)它。

順便說一句,為純虛函數(shù)提供一個定義并沒有被C++所禁止。也就是說,你可以為Shape::draw提供一套具體實現(xiàn),而C++不會報錯,但是在調(diào)用這種函數(shù)時,必須要加上類名:

Shape *ps = new Shape;             // 錯!Shape是抽象類

 

Shape *ps1 = new Rectangle;        // 正確

ps1->draw();                       // 調(diào)用Rectangle::draw

 

Shape *ps2 = new Ellipse;          // 正確

ps2->draw();                       // 調(diào)用Ellipse::draw

 

ps1->Shape::draw();                // 調(diào)用Shape::draw

 

ps2->Shape::draw();                // 調(diào)用 Shape::draw

上文所述的這一C++特征,除了作為你在雞尾酒會上的談資以外,似乎真正的用處很有限。然而就像你在下文中見到的一樣,這一特征也有一定的用武之地,它可以為簡單(非純)虛函數(shù)提供“超常安全”的默認具體實現(xiàn)。

簡單虛函數(shù)背后隱藏的內(nèi)情與純虛函數(shù)有些許不同。一般情況下,派生類繼承函數(shù)接口,但是簡單虛函數(shù)提供了一個具體實現(xiàn),派生類中可以覆蓋這一實現(xiàn)。如果你稍加思索,你就會發(fā)現(xiàn):

聲明簡單虛函數(shù)的目的就是:讓派生類繼承函數(shù)接口的同時繼承一個默認的具體實現(xiàn)

請觀察以下情形中的Shape::error

class Shape {

public:

  virtual void error(const std::string& msg);

  ...

};

接口要求每個類必須要提供一個error函數(shù),以便在程序出錯時調(diào)用,但是每個類都有適合自己的處理錯誤的方法。如果一個類并不想提供特殊的錯誤處理機制,那么它就可以返回調(diào)用Shape中提供的默認機制。也就是說,Shape::error的聲明就是告訴派生類的設(shè)計者,“你應(yīng)該提供一個error函數(shù),但是如果你不想自己編寫,那么也可以借助于Shape類的默認版本。”

實踐表明,允許簡單虛函數(shù)同時提供函數(shù)接口和默認實現(xiàn)是不安全的。至于原因,你可以設(shè)想一個XYZ航空公司的航班層次結(jié)構(gòu)。XYZ只有兩種飛機:A型和B型,它們飛行的航線是完全一致的。于是,XYZ這樣設(shè)計了層次結(jié)構(gòu):

class Airport { ... };             // 表示飛機場

 

class Airplane {

public:

  virtual void fly(const Airport& destination);

 

  ...

 

};

 

void Airplane::fly(const Airport& destination)

{

 默認代碼:使飛機抵達給定的目的地

}

 

class ModelA: public Airplane { ... };

 

class ModelB: public Airplane { ... };

此處Airplane::fly聲明為虛函數(shù),這是為了表明所有飛機必須要提供一個fly函數(shù),同時也基于以下事實:理論上講,不同型號的飛機需要提供不同版本的fly函數(shù)實現(xiàn)。然而,為了避免在ModelAModelB中出現(xiàn)同樣的代碼,我們將默認的飛行行為放置在Airplane::fly中,由ModelAModelB來繼承。

這是一個經(jīng)典的面向?qū)ο笤O(shè)計方案。當兩個類共享同一特征(即它們實現(xiàn)fly的方式)時,我們將這一共同特征移動到一個基類中,然后由兩個派生類來繼承這一共同特征。這一設(shè)計方案使得共同特征顯性化,避免了代碼重復(fù),為未來的更新工作提供了便利,減輕了長期維護的負擔——所有的一切都是面向?qū)ο蠹夹g(shù)極力倡導(dǎo)的。XYZ航空公司應(yīng)該感到十分驕傲了。

現(xiàn)在請設(shè)想:XYZ公司有了新的業(yè)務(wù)拓展,他們決定引進一款新型飛機——C型。C型飛機在某些方面與A型和B型有著本質(zhì)的區(qū)別,尤其是,C型飛機的飛行方式與前兩者完全不同。

XYZ的程序員將C型飛機添加進層次結(jié)構(gòu),但是由于他們急于讓新型飛機投入運營,他們忘記了重定義fly函數(shù):

class ModelC: public Airplane {

 

  ...                              // 沒有聲明任何fly函數(shù)

};

于是,在他們的代碼中將會遇到下面代碼中類似的問題:

Airport PDX(...);                  // PDX是我家附近一個飛機場

 

Airplane *pa = new ModelC;

 

...

 

pa->fly(PDX);                      // 調(diào)用了Airplane::fly

這將是一場災(zāi)難:因為此處做了一項可怕的嘗試,那就是讓ModelCModelAModelB的形式飛行。你將為這一嘗試付出慘痛的代價。

問題的癥結(jié)不在于Airplane::fly使用默認的行為,而是在沒有顯式說明的情況下ModelC需要繼承該行為的情況下就繼承了它。幸運的是以下這一點我們很容易做到:根據(jù)需要為派生類提供默認行為,如果派生類沒有顯式說明,那么就不為其提供。做到這一點的秘訣是:切斷虛函數(shù)的接口和默認實現(xiàn)之間的聯(lián)系。以下是一種實現(xiàn)方法:

class Airplane {

public:

  virtual void fly(const Airport& destination) = 0;

 

  ...

 

protected:

  void defaultFly(const Airport& destination);

};

 

void Airplane::defaultFly(const Airport& destination)

{

 默認代碼:使飛機抵達給定目的地

}

請注意這里的Airplane::fly是如何轉(zhuǎn)變成一個純虛函數(shù)的。它為飛行提供了接口。默認實現(xiàn)在Airplane類中也會出現(xiàn),但是現(xiàn)在它是以一個獨立函數(shù)的形式存在的——defaultFly。諸如ModelAModelB此類需要使用默認行為的類,只需要簡單地在它們的fly函數(shù)中內(nèi)聯(lián)調(diào)用defaultFly即可(請參見條目30中介紹的關(guān)于內(nèi)聯(lián)和虛函數(shù)之間的聯(lián)系):

class ModelA: public Airplane {

public:

  virtual void fly(const Airport& destination)

  { defaultFly(destination); }

 

  ...

};

 

class ModelB: public Airplane {

public:

  virtual void fly(const Airport& destination)

  { defaultFly(destination); }

 

  ...

};

對于ModelC類而言,繼承不恰當?shù)?span style="font-family:"Courier New";">fly實現(xiàn)是根本不可能的,因為Airplane中的純虛函數(shù)fly強制ModelC提供自己版本的fly

class ModelC: public Airplane {

public:

  virtual void fly(const Airport& destination);

 

  ...

};

 

void ModelC::fly(const Airport& destination)

{

 使C型飛機抵達目的地的代碼

}

這一方案亦非天衣無縫(程序員仍然會“復(fù)制/粘貼”出新的麻煩),但是它至少要比原始的設(shè)計方案更可靠。至于Airplane::defaultFly,由于此處它是Airplane及其派生類真實的實現(xiàn)。客戶只需要關(guān)注飛機可以飛行,而無須理會飛行功能是如何實現(xiàn)的。

Airplane::defaultFly是一個非虛函數(shù),這一點同樣重要。這是因為任何派生類都不應(yīng)該去重定義這一函數(shù),這是條目36所致力于講述的議題。如果defaultFly是虛函數(shù),那么你將會遇到一個遞歸的問題:如果一些派生類忘記了重定義defaultFly,那么它會怎樣呢?

類似于上文中介紹的flydefaultFly函數(shù),為接口和默認實現(xiàn)分別提供不同函數(shù)的方法,受到了一些人的質(zhì)疑。他們指出,盡管他們不懷疑將接口和默認實現(xiàn)分開處理的必要性,但是這樣做導(dǎo)致一些近親函數(shù)名字,從而污染了類名字空間。那么如何解決這一看上去自相矛盾的難題呢?我們知道純虛函數(shù)在具體的派生類中必須得到重新聲明,但是純虛函數(shù)自身也可以有具體實現(xiàn),借助這一點問題便迎刃而解。下面代碼中的Airplane層次結(jié)構(gòu)就利用了“純虛函數(shù)自身可以被定義”這一點:

class Airplane {

public:

  virtual void fly(const Airport& destination) = 0;

 

  ...

};

 

void Airplane::fly(const Airport& destination)

{                                  // 純虛函數(shù)的具體實現(xiàn)

 默認代碼:使飛機抵達給定的目的地

}

 

class ModelA: public Airplane {

public:

  virtual void fly(const Airport& destination)

  { Airplane::fly(destination); }

  ...

};

 

class ModelB: public Airplane {

public:

  virtual void fly(const Airport& destination)

  { Airplane::fly(destination); }

  ... 

};

 

class ModelC: public Airplane {

public:

  virtual void fly(const Airport& destination);

  ... 

};

 

void ModelC::fly(const Airport& destination)

{

 使C型飛機抵達目的地的代碼

}

這一設(shè)計方案與前一個幾乎是一致的。只是這里用純虛函數(shù)Airplane::fly代替了獨立函數(shù)Airplane::defaultFly。從本質(zhì)上講,這里的fly被分割成了兩個基本的組成部分,它的聲明確定它的接口,而它的定義確定它的默認行為(派生類可以使用這一定義,但只有在現(xiàn)實請求的前提下才可以)。然而將flydefaultFly合并起來,你就失去了將這兩個函數(shù)置于不同保護層次的能力:原先受保護的代碼(defaultFly中的代碼)現(xiàn)在是公共的了(因為這些代碼移動到了fly中)。

最后,讓我們把話題轉(zhuǎn)向Shape中的非虛函數(shù)——objectID

class Shape {

public:

  int objectID() const;

  ...

};

當一個成員函數(shù)不是虛函數(shù)時,你不應(yīng)該期待它會在不同的派生類中存在不同的行為。事實上,非虛成員函數(shù)確立了一個“個性化壁壘”,因為它確保了無論派生類在其他部分如何進行個性化,本函數(shù)所確定的行為不能被改變。也就是說:

聲明一個非虛函數(shù)的目的就是讓派生類繼承這一函數(shù)的接口,同時強制繼承其固定的具體實現(xiàn)。

你可以把shape::objectID的聲明想象成:每個Shape對象都有一個函數(shù)能生成“對象身份標識”的函數(shù)。這一“對象身份標識”總是以同一方式運行。這一方式由Shape::objectID的定義確定,任何派生類都不能償試更改這一方式。因為一個非虛函數(shù)就確定了一個“個性化壁壘”,所以在任何的派生類中都不允許重定義該函數(shù),這一點將在條目36中詳細講解。

純虛函數(shù),簡單虛函數(shù)和非虛函數(shù),不同的聲明方式使你能夠精確地指定你的派生類需要繼承什么:是僅僅繼承接口,還是同時繼承接口和實現(xiàn),抑或接口和一套固定的實現(xiàn)。由于這些不同種類的聲明意味著彼此在基礎(chǔ)層面存在著不同,因此你在聲明成員函數(shù)時,一定要仔細斟酌。如果你這樣做了,那么你將避免缺乏經(jīng)驗的類設(shè)計人員常犯的兩類錯誤:

首先,第一類錯誤是:將所有函數(shù)都聲明為非虛的。這樣做可以說斷送了派生類進行拓展的后路。非虛析構(gòu)函數(shù)更是陷阱重重(參見條目7)。當然,設(shè)計一個不需要作為基類的類無可厚非,這種情況下,清一色的一組非虛函數(shù)也是合乎情理的。然而,由于人們常常忽視虛函數(shù)和非虛函數(shù)之間的差異,還有對于“虛函數(shù)會對性能產(chǎn)生影響”的無端猜疑,導(dǎo)致我們的程序中充斥著過量的完全不包含虛函數(shù)的類。但事實上,幾乎每個需要充當基類的類都需要虛函數(shù)的支持。(同樣請參見條目7)

如果你談到虛函數(shù)的性能開銷問題,請允許我引用基于經(jīng)驗主義的“80-20法則”(同樣參見條目30),在一個典型的程序中,80%的運行時間將花費在20%的代碼上。這一法則十分重要,因為它意味著在一般情況下,80%的虛函數(shù)調(diào)用將不會對你的程序的整體性能造成任何影響。與其為虛函數(shù)是否會帶來無法承受的性能開銷而顧忌重重,還不如把精力放在程序中真正會帶來影響的那20%上。

另一個一般的問題是:將所有的成員函數(shù)都聲明為虛函數(shù)。有時候這么做是正確的——條目31中的接口類就是證據(jù)。然而,這樣做給人的印象就是這個類的設(shè)計者缺乏主心骨。在派生類中一些函數(shù)不應(yīng)該進行重定義,你必須要通過將這些函數(shù)聲明為非虛函數(shù)才能確保這一點。你應(yīng)該清楚,并不是讓客戶去重定義所有的函數(shù),你的類就成了萬能的了。如果你的類中包含個性化壁壘,那么就應(yīng)該大膽的將其聲明為非虛函數(shù)。


時刻牢記

接口繼承與實現(xiàn)繼承存在著不同。在公共繼承體系下,派生類總是繼承基類的接口。

純虛函數(shù)要求派生類僅繼承接口。

簡單(非純)虛函數(shù)要求派生類在繼承接口的同時繼承一個默認的實現(xiàn)。

非虛函數(shù)要求派生類繼承接口和強制固定內(nèi)容的實現(xiàn)。

posted on 2008-07-06 22:58 ★ROY★ 閱讀(2643) 評論(3)  編輯 收藏 引用 所屬分類: Effective C++

評論

# re: 【翻譯】[Effective C++中文版第3版][第34條]區(qū)分清接口繼承和實現(xiàn)繼承  回復(fù)  更多評論   

[quote]首先,第一類錯誤是:將所有函數(shù)都聲明為純虛的。這樣做可以說斷送了派生類進行拓展的后路。[/quote]
這句?
2008-07-14 17:06 | HUST

# re: 【翻譯】[Effective C++中文版第3版][第34條]區(qū)分清接口繼承和實現(xiàn)繼承  回復(fù)  更多評論   

支持!~
2008-10-13 15:18 | sandy

# re: 【翻譯】[Effective C++中文版第3版][第34條]區(qū)分清接口繼承和實現(xiàn)繼承  回復(fù)  更多評論   

第一類錯誤:將所有函數(shù)聲明為純虛函數(shù)。這樣做使得派生類完全沒有拓展的空間。
不明白。。
2012-05-23 19:27 | 恩戴米恩
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            在线看片欧美| 亚洲欧美大片| 亚洲美女av在线播放| 亚洲欧美日韩中文播放| 欧美一二三视频| 欧美一区二区三区电影在线观看| 久久高清国产| 欧美女人交a| 欧美不卡一区| 国产精品永久| 一本色道88久久加勒比精品| 91久久香蕉国产日韩欧美9色| 亚洲最新视频在线| 国产精品进线69影院| 亚洲欧洲日产国产综合网| 亚洲欧美中文日韩v在线观看| 亚洲欧洲日本一区二区三区| 欧美日韩中文字幕在线| 午夜激情亚洲| 欧美日韩在线一区| 亚洲天堂av在线免费观看| 亚洲高清电影| 国产精品乱子久久久久| 久久久久久久久久久久久9999| 欧美成人伊人久久综合网| 一区二区av| 日韩一级欧洲| 99视频一区二区三区| 精品91在线| 久久米奇亚洲| 黄色一区二区在线| 一区二区三区视频在线| 国产热re99久久6国产精品| 国产欧美一级| 亚洲一区二区三区中文字幕| 久久国产黑丝| 欧美不卡视频| 亚洲电影av| 一区二区三区四区国产| 亚洲激情视频在线| 99re66热这里只有精品3直播| 欧美久久99| 免费黄网站欧美| 在线观看日产精品| 午夜视频在线观看一区二区三区| 亚洲动漫精品| 亚洲小视频在线观看| 一区二区三区四区五区精品| 国产欧美视频一区二区| 久久中文欧美| 国内自拍亚洲| 欧美777四色影视在线| 美日韩精品视频| 国产日韩综合一区二区性色av| 久久天堂国产精品| 久久精品成人一区二区三区| 一区在线影院| 黑丝一区二区| 欧美一级欧美一级在线播放| 欧美在线视频不卡| 亚洲国产mv| 国产色爱av资源综合区| 久久9热精品视频| 老司机精品福利视频| 欧美亚洲综合在线| 久久精品日韩欧美| 先锋a资源在线看亚洲| 欧美日产在线观看| 久久久xxx| 亚洲国产精品毛片| 午夜精品视频在线| 欧美aⅴ一区二区三区视频| 亚洲免费视频在线观看| 久久香蕉国产线看观看网| 久久久一区二区三区| 另类欧美日韩国产在线| 久久av一区| 亚洲国产精品综合| 一区二区三区日韩在线观看| 亚洲欧美日韩国产成人精品影院| 一区二区欧美在线| 你懂的网址国产 欧美| 亚洲欧洲综合| 亚洲欧美在线磁力| 国产精品国产三级国产a| 亚洲韩国一区二区三区| 国产精品久久午夜| 国产日本欧美在线观看| 极品少妇一区二区| 欧美一区二区视频在线观看| 99视频精品在线| 亚洲图片欧美一区| 亚洲电影在线播放| 艳妇臀荡乳欲伦亚洲一区| 亚洲精品一区二区三区av| 久久九九国产| 欧美日韩中文精品| 国产精品美女久久久浪潮软件| 亚洲高清一区二| 麻豆精品精品国产自在97香蕉| 久久人人97超碰国产公开结果| 麻豆精品视频在线观看| 免费av成人在线| 久久久久在线| 欧美日韩少妇| 一片黄亚洲嫩模| 欧美国产一区视频在线观看 | 欧美网站大全在线观看| 一区二区三区高清不卡| 亚洲精品自在久久| 亚洲福利久久| 久久超碰97人人做人人爱| 亚洲天堂免费在线观看视频| 久久久久久久国产| 国产精品wwwwww| 欧美国产日本高清在线| 一区二区三欧美| 黑人巨大精品欧美一区二区| 美女亚洲精品| 久久久美女艺术照精彩视频福利播放 | 亚洲美女诱惑| 欧美日韩在线不卡一区| 亚洲午夜在线观看视频在线| 国内精品视频在线播放| 久久久精品欧美丰满| 欧美裸体一区二区三区| 亚洲大胆女人| 久久看片网站| 最新亚洲激情| 亚洲欧美在线免费观看| 国产精品久久久久久久久久直播| 亚洲一区二区三区国产| 亚洲激情av在线| 国产精品毛片va一区二区三区| 女生裸体视频一区二区三区| 欧美日韩高清在线一区| 国产一区二区三区在线观看免费| 久久国产精品99久久久久久老狼| 蜜臀久久99精品久久久画质超高清| 亚洲美女91| 欧美国产视频一区二区| 亚洲自拍高清| 亚洲欧美一区二区原创| 最近中文字幕mv在线一区二区三区四区 | 久久久久中文| 一本色道久久综合亚洲精品婷婷| 亚洲第一在线综合网站| 99视频有精品| 久久久在线视频| 农夫在线精品视频免费观看| 国产专区欧美专区| 午夜精品久久久久久99热| 你懂的亚洲视频| 另类亚洲自拍| 99精品国产热久久91蜜凸| 久久麻豆一区二区| 99综合精品| 亚洲国产成人精品视频| 亚洲人成网站在线播| 精品动漫3d一区二区三区免费版| 韩日欧美一区| 国产精品porn| 亚洲人成亚洲人成在线观看图片 | 欧美伊人久久久久久久久影院| 亚洲成色精品| 亚洲一区二区精品视频| 亚洲午夜小视频| 欧美激情第10页| 国产一区二区三区的电影 | 欧美日韩一区二区精品| 久久女同互慰一区二区三区| 最新精品在线| 亚洲欧美日韩中文在线制服| 牛牛影视久久网| 久久九九精品99国产精品| 在线亚洲精品| 亚洲一区二区精品在线| 亚洲欧美日本精品| 美女精品视频一区| 欧美日本一道本| 国语对白精品一区二区| 亚洲精品乱码久久久久久日本蜜臀| 亚洲福利视频一区二区| 午夜精品久久久久久久久久久久久 | 精品白丝av| 99国产精品久久久久久久成人热| 久久精视频免费在线久久完整在线看| 蜜臀av国产精品久久久久| 亚洲第一二三四五区| 性欧美1819sex性高清| 国产日本精品| 欧美亚洲免费| 亚洲伊人久久综合| 欧美人与性禽动交情品| 亚洲高清中文字幕| 久久av资源网站| 国内揄拍国内精品久久| 欧美二区乱c少妇| 欧美成人一区二区三区| 亚洲国产cao|