• <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>

            重載(overload),覆蓋(override),隱藏(hide)的區(qū)別

            寫(xiě)正題之前,先給出幾個(gè)關(guān)鍵字的中英文對(duì)照,重載(overload),覆蓋(override),隱藏(hide)。在早期的C++書(shū)籍中,可能翻譯的人不熟悉專(zhuān)業(yè)用語(yǔ)(也不能怪他們,他們不是搞計(jì)算機(jī)編程的,他們是英語(yǔ)專(zhuān)業(yè)的),常常把重載(overload)和覆蓋(override)搞錯(cuò)!

              我們先來(lái)看一些代碼及其編譯結(jié)果。

              實(shí)例一:
              #include "stdafx.h"
              #include <iostream.h>

              class CB
              {
              public:
            ?    void f(int)
            ?    {
            ?      ?cout << "CB::f(int)" << endl;
                ?}

              };


              class CD : public CB
              {
              public:
                ?void f(int,int)
            ?    {
            ??      cout << "CD::f(int,int)" << endl;
            ?    }

            ?    void test()
            ?    {
            ??     f(1);
            ?    }
              };

             int main(int argc, char* argv[])
             {
               ?return 0;
             }
            編譯了一下
            error C2660: 'f' : function does not take 1 parameters


            結(jié)論:在類(lèi)CD這個(gè)域中,沒(méi)有f(int)這樣的函數(shù),基類(lèi)中的void f(int)被隱藏

              如果把派生CD中成員函數(shù)void f(int,int)的聲明改成和基類(lèi)中一樣,即f(int),基類(lèi)中的void f(int)還是一樣被覆蓋,此時(shí)編譯不會(huì)出錯(cuò),在函數(shù)中test調(diào)用的是CD中的f(int) 

              所以,在基類(lèi)中的某些函數(shù),如果沒(méi)有virtral關(guān)鍵字,函數(shù)名是f(參數(shù)是什么我們不管),那么如果在派生類(lèi)CD中也聲明了某個(gè)f成員函數(shù),那么在類(lèi)CD域中,基類(lèi)中所有的那些f都被隱藏。
              如果你比較心急,想知道什么是隱藏,看文章最后的簡(jiǎn)單說(shuō)明,不過(guò)我建議你還是一步一步看下去。

              我們剛才說(shuō)的是沒(méi)有virtual的情況,如果有virtual的情況呢??
              實(shí)例二:

            #include "stdafx.h"
            #include <iostream.h>

            class CB
            {
            public:
            ?  virtual void f(int)
            ?  {
            ??    cout << "CB::f(int)" << endl;
            ?  }

            };


            class CD : public CB
            {
            public:
              ?void f(int)
            ?  {
            ??    cout << "CD::f(int)" << endl;
              ?}

            };

            int main(int argc, char* argv[])
            {
            ? return 0;
            }

              這么寫(xiě)當(dāng)然是沒(méi)問(wèn)題了,在這里我不多費(fèi)口舌了,這是很簡(jiǎn)單的,多態(tài),虛函數(shù),然后什么指向基類(lèi)的指針指向派生類(lèi)對(duì)象阿,通過(guò)引用調(diào)用虛函數(shù)阿什么的,屬性多的很咯,什么??你不明白??隨便找本C++的書(shū),對(duì)會(huì)講多態(tài)和虛函數(shù)機(jī)制的哦!!
              這種情況我們叫覆蓋(override)!覆蓋指的是派生類(lèi)的虛擬函數(shù)覆蓋了基類(lèi)的同名且參數(shù)相同的函數(shù)!
              在這里,我要強(qiáng)調(diào)的是,這種覆蓋,要滿(mǎn)足兩個(gè)條件
             (a)有virtual關(guān)鍵字,在基類(lèi)中函數(shù)聲明的時(shí)候加上就可以了
             (b)基類(lèi)CB中的函數(shù)和派生類(lèi)CD中的函數(shù)要一模一樣,什么叫一模一樣,函數(shù)名,參數(shù),返回類(lèi)型三個(gè)條件
              有人可能會(huì)對(duì)(b)中的說(shuō)法質(zhì)疑,說(shuō)返回類(lèi)型也要一樣??
              是,覆蓋的話必須一樣,我試了試,如果在基類(lèi)中,把f的聲明改成virtual int f(int),編譯出錯(cuò)了
              error C2555: 'CD::f' : overriding virtual function differs from 'CB::f' only by return type or calling convention
              所以,覆蓋的話,必須要滿(mǎn)足上述的(a)(b)條件

              那么如果基類(lèi)CB中的函數(shù)f有關(guān)鍵字virtual ,但是參數(shù)和派生類(lèi)CD中的函數(shù)f參數(shù)不一樣呢,
            實(shí)例三:
            ? #include "stdafx.h"
            #include <iostream.h>

            class CB
            {
             public:
            ?   virtual? void f(int)
              ?{
            ?    ?cout << "CB::f(int)" << endl;
              ?}

            }
            ;


            class CD : public CB
            {
            public:
            ?   void f(int,int)
            ?  {
            ?   ?cout << "CD::f(int,int)" << endl;
            ?  }

            ?  void test()
            ?  {
            ?    ?f(1);
            ?  }
            }
            ;

            int main(int argc, char* argv[])
            {
            ?return 0;
            }

            編譯出錯(cuò)了,
            ?error C2660: 'f' : function does not take 1 parameters
              咦??好面熟的錯(cuò)??對(duì),和實(shí)例一中的情況一樣哦,結(jié)論也是基類(lèi)中的函數(shù)被隱藏了。

              通過(guò)上面三個(gè)例子,得出一個(gè)簡(jiǎn)單的結(jié)論
            如果基類(lèi)中的函數(shù)和派生類(lèi)中的兩個(gè)名字一樣的函數(shù)f
            滿(mǎn)足下面的兩個(gè)條件
            (a)在基類(lèi)中函數(shù)聲明的時(shí)候有virtual關(guān)鍵字
            (b)基類(lèi)CB中的函數(shù)和派生類(lèi)CD中的函數(shù)一模一樣,函數(shù)名,參數(shù),返回類(lèi)型都一樣。
            那么這就是叫做覆蓋(override),這也就是虛函數(shù),多態(tài)的性質(zhì)

            那么其他的情況呢??只要名字一樣,不滿(mǎn)足上面覆蓋的條件,就是隱藏了。

            下面我要講最關(guān)鍵的地方了,好多人認(rèn)為,基類(lèi)CB中的f(int)會(huì)繼承下來(lái)和CD中的f(int,int)在派生類(lèi)CD中構(gòu)成重載,就像實(shí)例一中想像的那樣。
              對(duì)嗎?我們先看重載的定義
              重載(overload):
              必須在一個(gè)域中,函數(shù)名稱(chēng)相同但是函數(shù)參數(shù)不同,重載的作用就是同一個(gè)函數(shù)有不同的行為,因此不是在一個(gè)域中的函數(shù)是無(wú)法構(gòu)成重載的,這個(gè)是重載的重要特征
              必須在一個(gè)域中,而繼承明顯是在兩個(gè)類(lèi)中了哦,所以上面的想法是不成立的,我們測(cè)試的結(jié)構(gòu)也是這樣,派生類(lèi)中的f(int,int)把基類(lèi)中的f(int)隱藏了
              所以,相同的函數(shù)名的函數(shù),在基類(lèi)和派生類(lèi)中的關(guān)系只能是覆蓋或者隱藏。

              在文章中,我把重載和覆蓋的定義都給了出來(lái)了,但是一直沒(méi)有給隱藏的定義,在最后,我把他給出來(lái),這段話是網(wǎng)上google來(lái)的,比較長(zhǎng),你可以簡(jiǎn)單的理解成,在派生類(lèi)域中,看不到基類(lèi)中的那個(gè)同名函數(shù)了,或者說(shuō),是并沒(méi)有繼承下來(lái)給你用,呵呵,如實(shí)例一 那樣。
              

            隱藏(hide):
            指的是派生類(lèi)的成員函數(shù)隱藏了基類(lèi)函數(shù)的成員函數(shù).隱藏一詞可以這么理解:在調(diào)用一個(gè)類(lèi)的成員函數(shù)的時(shí)候,編譯器會(huì)沿著類(lèi)的繼承鏈逐級(jí)的向上查找函數(shù)的定義,如果找到了那么就停止查找了,所以如果一個(gè)派生類(lèi)和一個(gè)基類(lèi)都有同一個(gè)同名(暫且不論參數(shù)是否相同)的函數(shù),而編譯器最終選擇了在派生類(lèi)中的函數(shù),那么我們就說(shuō)這個(gè)派生類(lèi)的成員函數(shù)"隱藏"了基類(lèi)的成員函數(shù),也就是說(shuō)它阻止了編譯器繼續(xù)向上查找函數(shù)的定義.

            posted on 2007-03-12 21:47 永遇樂(lè) 閱讀(5992) 評(píng)論(6)  編輯 收藏 引用 所屬分類(lèi): C & C++

            評(píng)論

            # re: 重載(overload),覆蓋(override),隱藏(hide)的區(qū)別 2009-05-22 17:15 宋濤

            大哥,你的blog做得太漂亮了。  回復(fù)  更多評(píng)論   

            # re: 重載(overload),覆蓋(override),隱藏(hide)的區(qū)別 2013-09-27 20:05 vergil125

            說(shuō)的很好  回復(fù)  更多評(píng)論   

            # re: 重載(overload),覆蓋(override),隱藏(hide)的區(qū)別[未登錄](méi) 2013-12-04 09:03


              回復(fù)  更多評(píng)論   

            # re: 重載(overload),覆蓋(override),隱藏(hide)的區(qū)別 2014-03-22 09:33 cheering

            好,轉(zhuǎn)載之  回復(fù)  更多評(píng)論   

            # re: 重載(overload),覆蓋(override),隱藏(hide)的區(qū)別[未登錄](méi) 2014-05-24 21:51 twlkyao

            講解的很詳細(xì),受教了。  回復(fù)  更多評(píng)論   

            # re: 重載(overload),覆蓋(override),隱藏(hide)的區(qū)別[未登錄](méi) 2016-01-06 22:47 Name

            f找不到,從本質(zhì)上來(lái)講是因?yàn)樵谶@個(gè)作用域中找不到這樣的f形式。這就跟你在for循環(huán)中定義了int a,那么for循環(huán)外的a你沒(méi)有辦法使用一樣。你想使用f就可以加上作用域CB::f(1)

            override中,返回類(lèi)型在一種情況下可以不一模一樣,如果基類(lèi)中返回CB*,派生類(lèi)返回的CD*,而且是公有繼承,這時(shí)候雖然返回類(lèi)型不同,但是仍然是override  回復(fù)  更多評(píng)論   

            <2025年5月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            導(dǎo)航

            統(tǒng)計(jì)

            常用鏈接

            留言簿(6)

            隨筆分類(lèi)

            推薦Blog

            友情鏈接

            搜索

            最新評(píng)論

            閱讀排行榜

            一本久久免费视频| 久久香蕉综合色一综合色88| 99久久人妻无码精品系列| 亚洲人成精品久久久久| 亚洲人成无码网站久久99热国产| 成人亚洲欧美久久久久| 品成人欧美大片久久国产欧美... 品成人欧美大片久久国产欧美 | 久久美女人爽女人爽| 精品久久人人爽天天玩人人妻| 久久人人爽人人爽人人片AV麻豆| 久久99精品久久久久久齐齐| 99久久亚洲综合精品网站| 国产精品成人99久久久久| 91久久香蕉国产熟女线看| 久久国产精品二国产精品| 三级韩国一区久久二区综合| 久久婷婷五月综合成人D啪| 无码国内精品久久人妻蜜桃| 国产国产成人精品久久| 国内精品久久久久久久久| 久久亚洲精品无码播放| 国产免费久久精品99re丫y| 人妻少妇久久中文字幕一区二区 | 亚洲AV成人无码久久精品老人| 99精品国产综合久久久久五月天| 奇米影视7777久久精品| 91精品国产91久久| 久久亚洲精品无码aⅴ大香| 色婷婷久久综合中文久久蜜桃av| 久久本道伊人久久| 一级a性色生活片久久无少妇一级婬片免费放| 中文国产成人精品久久亚洲精品AⅤ无码精品 | 久久精品国产日本波多野结衣| 色88久久久久高潮综合影院| 99久久精品免费观看国产| 中文字幕人妻色偷偷久久| 婷婷综合久久中文字幕| 久久久久久久波多野结衣高潮| 久久精品视频网| 久久精品成人欧美大片| 久久乐国产精品亚洲综合|