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

洛譯小筑

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

[ECPP讀書筆記 條目33] 避免隱藏繼承而來的名字

莎士比亞對于“名字”有著獨特的見解。“名字意味著什么?玫瑰不叫玫瑰,依然芬芳如故。”大師還寫道:“倘若有人偷竊了我的好名字……事實上會讓我變得一貧如洗。”讓這兩段名句引領(lǐng)我們?nèi)ヌ骄緾++中繼承的名字。

事實上,本節(jié)討論的問題與繼承并沒有太大關(guān)系。它僅僅關(guān)系到作用域。我們都能讀懂下面的代碼:

int x;                             // 全局變量

 

void someFunc()

{

  double x;                        // 局部變量

 

  std::cin >> x;                   // 讀一個新值賦給局部變量x

}

為x賦值的語句是關(guān)于局部變量x的,而不是全局變量x,這是因為內(nèi)部作用域隱藏了(“遮擋了”)外部作用域的名字。我們可以將這種域間狀況用下圖描述:


當(dāng)編譯器執(zhí)行至someFunc的作用域內(nèi)并且遇到名字x時,它將在局部作用域內(nèi)查找,以便確認(rèn)此處是否包含與x這個名字相關(guān)的操作。因為如果有的話,編譯器就不會再去檢查其它任何作用域了。在這上面的示例中,someFunc中的xdouble類型的,全局變量xint類型的,但是這無關(guān)緊要,C++的名字隱藏準(zhǔn)則只會做一件事情:隱藏名字。至于名字相關(guān)的類型是否一致并不重要。本例中,double類型的x隱藏了int類型的x

引入繼承。我們知道當(dāng)我們在一個派生類的成員函數(shù)中企圖引用基類的某些內(nèi)容(比如成員函數(shù)、typedef、或者數(shù)據(jù)成員等等)時,編譯器能夠找出我們所引用的內(nèi)容,因為派生類所繼承的內(nèi)容在基類中都做過聲明。這里真正的工作方式實際上是:派生類的作用域嵌套在基類的作用域中。請看下面示例:

class Base {

private:

  int x;

 

public:

  virtual void mf1() = 0;

  virtual void mf2();

  void mf3();

 

  ...

};

 

class Derived: public Base {

public:

  virtual void mf1();

  void mf4();

 

  ...

};


本示例中同時存在公共的、私有的名字,另外同時包含了數(shù)據(jù)成員和成員函數(shù)的名字。成員函數(shù)還包括純虛函數(shù)、簡單虛函數(shù)(非純虛的)和非虛函數(shù)。這就是向大家強(qiáng)調(diào),我們此處討論的中心話題就是名字。示例中還可以添加類型的名字,比如枚舉類型、嵌套類以及預(yù)定義類型的名字。這里討論的核心是:它們都是名字,而它們是為哪些東西命名的并不重要。示例中使用了單一繼承結(jié)構(gòu),然而一旦你了解了C++中單一繼承的行為方式之后,多重繼承的行為也就不難推斷了。

假定繼承類中mf4是這樣實現(xiàn)的(部分內(nèi)容):

void Derived::mf4()

  ...

  mf2();

  ...

}

當(dāng)編譯器看到這個函數(shù)中使用了mf2這個名字,它就能夠找到mf2的出處。編譯器是這樣做到的:它通過搜尋名字為mf2的那處聲明所在的作用域。首先它在本地作用域(也就是mf4以內(nèi))查找,但是沒有找到任何名字為mf2的聲明。隨后編譯器搜尋當(dāng)前包含它的域,也就是Derived類的作用域。仍然沒有找到,于是又轉(zhuǎn)向搜索上一層作用域,也就是基類。在這里編譯器終于找到了名叫mf2的東西,于是搜索結(jié)束。如果Base類中依然沒有mf2,那么搜索仍會繼續(xù),從包含Base的名字空間開始,到全局作用域為止。

雖然我剛剛描述的查找過程是精確的,但是其對于C++中名字查找機(jī)制的描述依然沒有做到面面俱到。所幸我們的目標(biāo)并不是對名字查找機(jī)制刨根問底從而去編寫一個編譯器。我們的目標(biāo)是避免惱人的意外發(fā)生,針對這一點,我們掌握的信息已經(jīng)足夠了。

請再次考慮上面的示例,這次我們做一些小的改動:為mf1mf3個添加一個重載版本,并且在Derived中為mf3添加一個新版本。(如條目36所講,Derived中重載版本的mf3(一個繼承而來的非虛函數(shù))將會使這樣的設(shè)計存在無法避免的潛在危險,但是在此問題的焦點是繼承下名字的可見性,我們暫且忽略這一問題。)

class Base {

private:

  int x;

 

public:

  virtual void mf1() = 0;

  virtual void mf1(int);

 

  virtual void mf2();

 

  void mf3();

  void mf3(double);

  ...

};

 

class Derived: public Base {

public:

  virtual void mf1();

  void mf3();

  void mf4();

  ...

};


這段代碼的行為將會使每個乍看到它的C++程序員吃上一驚。由于基于作用域的名字隱藏機(jī)制并沒有改變,因此基類中所有名叫mf1mf3的函數(shù)都被派生類中的mf1mf3所隱藏。從名字查找的角度看,Base::mf1Base::mf3不再被Derived繼承!

Derived d;

int x;

 

...

d.mf1();                   // 正確,調(diào)用Derived::mf1

d.mf1(x);                  // 錯誤! Derived::mf1隱藏了Base::mf1

d.mf2();                   // 正確,調(diào)用Base::mf2

 

d.mf3();                   // 正確,調(diào)用Derived::mf3

d.mf3(x);                  // 錯誤!Derived::mf3隱藏了Base::mf3

就像你所看到的,即使同一函數(shù)在基類和派生類中的參數(shù)表不同,基類中該函數(shù)依然會被隱藏,而且這一結(jié)論不會因函數(shù)是否為虛函數(shù)而改變。在本條目最開端的示例中,someFunc中的double x隱藏了全局的int x,此處的情況類似,Derived類中的函數(shù)mf3也會將Base類中名叫mf3但類型不同的函數(shù)隱藏起來。

C++這一特性的理論基礎(chǔ)是:可以防止一類繼承意外的發(fā)生,那就是當(dāng)你為一個庫或應(yīng)用框架創(chuàng)建一個新的派生類時,你可能會去繼承遠(yuǎn)族基類中的重載版本。遺憾的是,我們通常情況下恰恰希望這么做。事實上,如果你使用公共繼承,但不繼承重載的元素,那么就有悖于公共繼承的一項基本原則——基類和派生類之間是“Derived是一個Base”關(guān)系(見條目32)。既然如此,你就需要時時刻刻重載C++默認(rèn)情況下隱藏的繼承而來的名字。

這一工作通過使用using聲明來實現(xiàn):

class Base {

private:

  int x;

 

public:

  virtual void mf1() = 0;

  virtual void mf1(int);

 

  virtual void mf2();

 

  void mf3();

  void mf3(double);

  ...

};

 

class Derived: public Base {

public:

  using Base::mf1;       // 讓基類中所有名為mf1mf3的東西

  using Base::mf3;       // Derived的作用域中可見(并且是公有的)

 

  virtual void mf1();

  void mf3();

  void mf4();

  ...

};


現(xiàn)在,繼承將按部就班進(jìn)行:

Derived d;

int x;

 

...

 

d.mf1();               // 依然正常,依然調(diào)用Derived::mf1

d.mf1(x);              // 現(xiàn)在可以了,調(diào)用了Base::mf1

 

d.mf2();               // 依然正常,依然調(diào)用Derived::mf1

 

d.mf3();               // 正常,調(diào)用Derived::mf3

d.mf3(x);              // 現(xiàn)在可以了,調(diào)用了Base::mf3

                       // (此處的xint隱式轉(zhuǎn)換為double

                       // 從而使Base::mf3的調(diào)用合法。)

這意味著如果你繼承一個包含重載函數(shù)的基類,并且你僅期望對其中一部分進(jìn)行重定義或重載,你就應(yīng)該為每一個不期望被隱藏的名字添加一條using聲明。如果你不這樣做,一些你希望繼承下來的名字將可能被隱藏。

不難想象,某些場合你可能不想把基類中所有的函數(shù)繼承下來。但是在公共繼承體系下這是無論如何不可行的,再次聲明,這是違背公共繼承“Derived是一個Base”關(guān)系的。(這也是為什么上文中using聲明要置于派生類中的公共元素部分:因為基類中公有的名字在公共派生類中必須是公有的。)然而在私有繼承體系下(參見條目39),這種不完全繼承在某些情況下是有意義的。比如,假設(shè)Derived類私有繼承自Base,并且Derived只希望繼承mf1不包含參數(shù)的那個版本。using聲明在此就不會奏效了,因為它將使該名字所代表的所有繼承版本的函數(shù)在派生類中可見。在這種情況下可以使用另一種技術(shù),我們稱之為“轉(zhuǎn)發(fā)函數(shù)”:

class Base {

public:

  virtual void mf1() = 0;

  virtual void mf1(int);

 

  ...                              // 同上

};

 

class Derived: private Base {

public:

  virtual void mf1()               // 轉(zhuǎn)發(fā)函數(shù)

  { Base::mf1(); }                 // 隱式內(nèi)聯(lián)(參見條目30

  ...                              // (關(guān)于對純虛函數(shù)的調(diào)用,請參見條目34

};

 

...

 

Derived d;

int x;

 

d.mf1();                           // 正常,調(diào)用Derived::mf1

d.mf1(x);                          // 錯誤!Base::mf1()被隱藏了

內(nèi)聯(lián)轉(zhuǎn)發(fā)函數(shù)的另一個用途是:在使用古老的編譯器時,它們通常不支持使用using聲明來為繼承類的作用域引入繼承的名字(這實際上是編譯器的缺陷)。此時可以使用內(nèi)聯(lián)轉(zhuǎn)發(fā)函數(shù)。

以上是繼承和名字隱藏的全部內(nèi)容,但是如果繼承涉及模板,那么我們將以另一種形式面對“繼承而來的名字被隱藏”這一問題。對于模板繼承的全部細(xì)節(jié),請參見條目43。

時刻牢記

派生類中的名字會將基類中的名字隱藏起來。在公共繼承體系下,這是我們永遠(yuǎn)不希望見到的。

為了讓被隱藏名字再次可見,可以使用using聲明或者轉(zhuǎn)發(fā)函數(shù)。

posted on 2008-05-01 01:11 ★ROY★ 閱讀(2754) 評論(2)  編輯 收藏 引用 所屬分類: Effective C++

評論

# re: 【讀書筆記】[Effective C++第3版][第33條] 防止隱藏繼承的名字  回復(fù)  更多評論   

哈哈,是的,但是C#中不會影藏!
2008-05-04 08:56 | 夢在天涯

# re: 【讀書筆記】[Effective C++第3版][第33條] 防止隱藏繼承的名字  回復(fù)  更多評論   

“ l 派生類中的名字會將基類中的名字隱藏起來。在公有繼承體系下,這是我們所不希望見到的。”

---------------------------
使用PCLINT應(yīng)該可以檢查出來吧~~建議大家寫完代碼林特林特~~
2008-05-10 10:27 | zhang某人
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲精品美女久久久久| 欧美顶级艳妇交换群宴| 另类酷文…触手系列精品集v1小说| 午夜久久美女| 久久久精品免费视频| 久久久久久久久久久久久女国产乱| 久久久久久久久久久成人| 蜜桃av一区| 亚洲精选久久| 欧美亚洲午夜视频在线观看| 久久精品亚洲国产奇米99| 米奇777超碰欧美日韩亚洲| 欧美日韩天堂| 影音先锋亚洲精品| 亚洲午夜在线视频| 久久偷看各类wc女厕嘘嘘偷窃| 欧美高清自拍一区| 亚洲欧美国产高清| 欧美激情导航| 国产专区欧美专区| 99综合在线| 久久夜色精品| 亚洲午夜一区二区三区| 亚洲午夜一二三区视频| 黑人巨大精品欧美一区二区| 亚洲国产精品va在线观看黑人| 亚洲午夜黄色| 六十路精品视频| 亚洲天堂久久| 欧美成人精品在线视频| 国产亚洲精品一区二555| 日韩一级在线观看| 麻豆精品视频在线观看| 一区二区激情| 欧美激情一级片一区二区| 国内精品久久久久久久影视麻豆 | 亚洲视频www| 欧美 日韩 国产 一区| 国产精品一区免费观看| 亚洲另类在线一区| 免费观看日韩| 欧美一区二区三区四区在线观看| 欧美激情影院| 亚洲高清视频在线观看| 久久久久久色| 香蕉尹人综合在线观看| 欧美视频精品在线| 日韩午夜在线电影| 欧美黄色成人网| 另类天堂视频在线观看| 国产亚洲午夜| 久久er99精品| 亚洲欧美色婷婷| 国产日本欧美在线观看| 亚洲欧美日韩精品久久久久| 日韩午夜在线播放| 欧美日韩中文字幕| 亚洲欧美日韩第一区| 亚洲特色特黄| 国产欧美日韩不卡免费| 欧美一区午夜精品| 亚洲伊人网站| 国产一区二区精品在线观看| 久久久久久久久岛国免费| 欧美专区18| 亚洲国产高清aⅴ视频| 久久久免费av| 久久综合中文字幕| 亚洲另类一区二区| 日韩视频在线观看免费| 欧美日韩亚洲综合一区| 亚洲欧美久久久| 欧美亚洲视频一区二区| 国产一区二区精品在线观看| 久久综合色8888| 亚洲香蕉在线观看| 一区二区三区你懂的| 亚洲精品永久免费精品| 欧美日韩国产在线一区| 亚洲欧美成人网| 午夜精品久久久久影视| 狠狠色丁香婷婷综合| 亚洲成人在线视频网站| 欧美欧美天天天天操| 国产精品99久久久久久www| 亚洲一区一卡| 亚洲国产精品久久人人爱蜜臀 | 久久久久久久久久久久久女国产乱 | 亚洲人成在线观看网站高清| 欧美日韩在线不卡一区| 久久久99国产精品免费| 欧美va亚洲va香蕉在线| 亚洲欧美日韩在线不卡| 久久九九99视频| 一区二区三区日韩在线观看| 亚洲欧美日韩视频二区| 亚洲人成网站色ww在线| 亚洲欧美国产一区二区三区| 亚洲国产成人在线播放| 午夜精品久久| 一本一本久久a久久精品综合麻豆 一本一本久久a久久精品牛牛影视 | 亚洲精品1234| 国产精品亚洲综合一区在线观看 | 狠狠色丁香婷综合久久| 99精品免费视频| 在线播放亚洲一区| 亚洲网站啪啪| 一区二区三区国产在线观看| 久久久www| 欧美在线一二三区| 欧美日韩另类国产亚洲欧美一级| 久久人人爽人人爽| 国产精品嫩草99av在线| 亚洲第一精品电影| 一区精品在线| 欧美在线观看www| 亚洲欧美视频一区| 欧美午夜电影一区| 亚洲黄色在线视频| 尤物在线观看一区| 久久久xxx| 久久日韩粉嫩一区二区三区| 国产精品综合| 亚洲一区在线看| 午夜激情一区| 国产精品呻吟| 91久久精品国产91久久性色| 久久精品一区二区三区不卡| 极品av少妇一区二区| 亚洲欧美日韩电影| 亚洲综合精品自拍| 欧美三级在线| 一区二区欧美在线观看| av不卡在线观看| 欧美高清在线| 亚洲狠狠婷婷| 一区二区欧美日韩视频| 欧美精品激情在线| 亚洲精品欧洲| 亚洲伊人久久综合| 国产精品美女一区二区| 亚洲一区二区三区四区五区黄| 亚洲一区尤物| 国产精品永久免费视频| 亚洲视频在线观看免费| 欧美一区二视频| 狠狠色综合一区二区| 久热综合在线亚洲精品| 亚洲国产视频一区| 一区二区三区日韩在线观看| 国产精品ⅴa在线观看h| 亚洲永久免费视频| 久久米奇亚洲| 亚洲精品乱码久久久久久蜜桃91 | 亚洲免费视频网站| 久久亚洲不卡| 日韩一区二区免费高清| 国产精品美女www爽爽爽| 久久www免费人成看片高清| 免费不卡视频| 一区二区免费在线观看| 国产精品一区=区| 久久久www免费人成黑人精品| 欧美成人一区二区三区| 一区二区三区欧美成人| 国产无遮挡一区二区三区毛片日本| 久久久欧美一区二区| 亚洲免费观看视频| 久久久久在线观看| 中文精品视频| 伊人激情综合| 欧美性做爰猛烈叫床潮| 久久精品国产免费看久久精品| 亚洲第一在线综合网站| 亚洲欧美日韩电影| 亚洲高清不卡在线观看| 欧美日韩免费高清| 久久视频国产精品免费视频在线 | 毛片精品免费在线观看| 亚洲一区二三| 亚洲精品国产视频| 久久久久久夜精品精品免费| 亚洲乱码国产乱码精品精可以看| 国产精品久久久久秋霞鲁丝| 久久久视频精品| 亚洲制服丝袜在线| 亚洲美女精品一区| 欧美电影在线| 久久精品国语| 亚洲欧美日韩精品| 在线中文字幕一区| 亚洲黄网站在线观看| 国产伪娘ts一区| 国产精品久久久久三级| 欧美精品一区二区三区在线播放| 久久成人免费网| 久久女同互慰一区二区三区| 欧美性猛交99久久久久99按摩 | 久久精品亚洲乱码伦伦中文| 亚洲天堂偷拍|