demibug

            BlogJava :: 首頁(yè) :: 聯(lián)系 :: 聚合  :: 管理
            24 Posts :: 3 Stories :: 2 Comments :: 0 Trackbacks
          最近看<<高質(zhì)量C++>>時(shí)讀到的關(guān)于成員函數(shù)的重載/覆蓋/隱藏,把我的一點(diǎn)理解寫出來(lái),希望大家批評(píng)與指正.

          1. 重載、覆蓋與隱藏

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

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

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

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

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

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

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

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

          C.參數(shù)相同

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

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

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

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

           

          2.看下面這個(gè)例子代碼:

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

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

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

           

          4.總結(jié)

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


           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 2013-03-10 20:06 Hiji 閱讀(217) 評(píng)論(0)  編輯  收藏

          只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。


          網(wǎng)站導(dǎo)航:
           
          主站蜘蛛池模板: 陈巴尔虎旗| 江城| 南溪县| 文成县| 黎川县| 育儿| 嘉禾县| 嵊州市| 凤庆县| 定兴县| 鹤峰县| 英吉沙县| 瑞昌市| 资源县| 中方县| 镇坪县| 永靖县| 稻城县| 大埔县| 永宁县| 江川县| 开平市| 鹤峰县| 新绛县| 昭苏县| 文山县| 嘉兴市| 通海县| 宁波市| 疏勒县| 顺平县| 定南县| 贵德县| 安陆市| 武清区| 广昌县| 沈丘县| 绵竹市| 舟曲县| 玛纳斯县| 遵义县|