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

            平凡的世界

            神鷹忽展翅,頭頂青天飛
            隨筆 - 10, 文章 - 0, 評論 - 34, 引用 - 0
            數(shù)據(jù)加載中……

            關(guān)于C++類成員函數(shù)的重載、覆蓋、隱藏與virtual關(guān)鍵字

            關(guān)于C++類成員函數(shù)的重載、覆蓋、隱藏與virtual關(guān)鍵字

            Phoenix(phoenix8848@gmail.com)

              最近看<<高質(zhì)量C++>>時讀到的關(guān)于成員函數(shù)的重載/覆蓋/隱藏,把我的一點理解寫出來,希望大家批評與指正.

            1. 重載、覆蓋與隱藏

            1).重載:成員函數(shù)具有以下的特征時發(fā)生“重載”

            A.相同的范圍(同一個類中)

            B.函數(shù)的名字相同

            C.參數(shù)類型不同(不能進(jìn)行隱式類型轉(zhuǎn)換)

            DVirtual關(guān)鍵字可有可無

            2).覆蓋(也叫“繼承”):指派生類函數(shù)覆蓋基類函數(shù),特征是:

            A.不同的范圍(分別位于基類與派生類中)

            B.函數(shù)名字相同

            C.參數(shù)相同

            D.基類函數(shù)必須有virtual關(guān)鍵字

            3).隱藏:是指派生類的函數(shù)屏蔽了與其同名的基類函數(shù),規(guī)則如下:

            A.如果派生類的函數(shù)與基類的函數(shù)同名,但是參數(shù)不同,此時不論有無virtual關(guān)鍵字,基類的函數(shù)都將被隱藏,注意別與重載混淆)

            B.如果派生類的函數(shù)與基類的函數(shù)同名,并且參數(shù)也相同,但是基類函數(shù)沒有virtual關(guān)鍵字,此時基類的函數(shù)被隱藏(注意別與覆蓋混淆)

             

            2.看下面這個例子代碼:

             1 #include <iostream>
             2 using std::cout;
             3 using std::endl;
             4 
             5 class Base
             6 {
             7 public:
             8     virtual void f(float x){ cout << "Base::f(float) " << x << endl;}
             9     void g(float x){ std::cout << "Base::g(float) " << x << std::endl;}
            10     void h(float x){ std::cout << "Base::h(float) " << x <<std::endl;}
            11 };
            12 
            13 class Derived : public Base
            14 {
            15 public:
            16     virtual void f(float x){ std::cout << "Derived::f(float) " << x << std::endl;}
            17     void g(int x){ std::cout << "Derived::g(int) " << x << std::endl;}
            18     void h(float x){ std::cout << "Derived::h(float) " << x << std::endl;}
            19 };
            20 
            21 void main(void)
            22 {
            23     Derived d;
            24     Base *pb = &d;
            25     Derived *pd = &d;
            26 
            27     pb->f(3.14f);//Derived::f(float) 3.14
            28     pd->f(3.14f);//Derived::f(float) 3.14
            29 
            30     pb->g(3.14f);//Base::g(float) 3.14
            31     pd->g(3.14f);//Derived::g(int) 3
            32 
            33     pb->h(3.14f);//Base:h(float) 3.14
            34     pd->h(3.14f);//Derived::h(float) 3.14
            35 }

             

            3. 解釋

            2728行,派生類的Derived::f(float x)通過virtual關(guān)鍵字繼承(覆蓋)了基類的Base::f(float x)方法,所以這里無論采有基類指針還是派生類指針,最后調(diào)用的其實都是Derived::f(float x)方法。這正是一般情況我們所期望的。

            30行,由于基類的Base::g()沒有用virtual關(guān)鍵字聲明,所以這里它不會被派生類的Derived::g()方法覆蓋。所以通過基類指針訪問時只能訪問到Base::g(float x),而在31行通過派生類指針時可以訪問的方法有Base::g(float x)Derived::g(int x),這兩個方法雖然方法名相同而且參數(shù)不同(似乎)符合重載的標(biāo)準(zhǔn),但是它們卻分屬于不同的“域”因此重載不會發(fā)生,這時Derived::g(int x)就只能把Base::g(float x)“隱藏”掉。

            同上,在第33行通過基類指針能訪問的方法只有Base::h(float x),由于該方法沒有被virtual關(guān)鍵字聲明,所以不會被派生類方法Derived::h(float x)“替換”,因此調(diào)用的是Base::h(float x)。而在第34行通過派生類指針可以訪問的方法同時有Base::h(float x)Derived::h(float x),這似乎又沖突,而這時C++的“隱藏”規(guī)則發(fā)生作用,所以派生類方法Derived::h(float x)把基類方法Base::h(float x)“隱藏”,于是Derived::h(float x)被調(diào)用。

             

            4.總結(jié)

                   C++的“重載”、“繼承”與“隱藏”機制比一般想象中的要復(fù)雜,而這就突顯了virtual關(guān)鍵字的重要性。所以在派生類存在的前提下一,一定要把基類中可能在派生類中也實現(xiàn)的方法用virtual關(guān)鍵字聲明。除非在特殊情況下,比如需要檢查指針類型的時候。


             1 #include <iostream>
             2 using std::cout;
             3 using std::endl;
             4 
             5 class Base
             6 {
             7 public:
             8     void CheckType(void){ cout << "This's Base Ptr" << endl;}
             9 };
            10 
            11 class Derived : public Base
            12 {
            13 public:
            14     void CheckType(void){ cout << "This;s Derived Ptr" << endl;}
            15 };
            16 
            17 void main(void)
            18 {
            19     Derived d;
            20     Base *pb = &d;
            21     Derived *pd = &d;
            22 
            23     pb->CheckType();//This's Base Ptr
            24     pd->CheckType();//This's Derived Ptr
            25 }
            26 

             

            posted on 2008-10-13 07:42 西門有悔 閱讀(6584) 評論(1)  編輯 收藏 引用

            評論

            # re: 關(guān)于C++類成員函數(shù)的重載、覆蓋、隱藏與virtual關(guān)鍵字  回復(fù)  更多評論   

            整理得非常好
            不過對隱藏的概念應(yīng)該還可以簡化為:基類成員函數(shù)中,不滿足覆蓋條件的派生類同名成員函數(shù),都視為隱藏(既基類方法不能被派生類繼承使用)。

            而顯然滿足覆蓋的充要條件是:
            (a)在基類中函數(shù)聲明的時候有virtual關(guān)鍵字
            (b)基類中的函數(shù)和派生類中的函數(shù)聲明一模一樣即函數(shù)名,參數(shù),返回類型都一樣
            2008-10-16 14:58 | frank.sunny
            久久Av无码精品人妻系列| 国产巨作麻豆欧美亚洲综合久久 | 亚洲国产精品无码久久一线 | 日日狠狠久久偷偷色综合免费| 亚洲国产成人精品女人久久久 | 伊人久久精品无码二区麻豆| 无码精品久久久久久人妻中字| 久久精品人人做人人爽97| 中文字幕成人精品久久不卡| 性做久久久久久免费观看| 亚洲成色WWW久久网站| 国产精品午夜久久| 18岁日韩内射颜射午夜久久成人| 久久99国产精一区二区三区| 亚洲精品国产第一综合99久久| 青青青青久久精品国产h| 久久久久人妻一区精品 | 香蕉久久夜色精品国产2020| 国产精品九九九久久九九| 亚洲国产精品无码久久青草| 99久久中文字幕| 中文字幕精品久久| 亚洲国产精品婷婷久久| 欧洲人妻丰满av无码久久不卡| 国产成人精品久久亚洲| 国内精品久久久久影院一蜜桃| 色99久久久久高潮综合影院| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 午夜视频久久久久一区 | 久久人与动人物a级毛片| 国产精品成人久久久久三级午夜电影 | 久久A级毛片免费观看| 国内精品伊人久久久久777| 一本色道久久88综合日韩精品 | 国产精品一区二区久久| 久久久久久久人妻无码中文字幕爆| 青青草国产97免久久费观看| 久久人人超碰精品CAOPOREN| 久久国产精品-国产精品| 久久精品国产亚洲av麻豆小说 | 欧美无乱码久久久免费午夜一区二区三区中文字幕|