C++ : 重載、覆蓋、和隱藏

 這幾個概念都有一個共同點:函數名稱相同,所以不免讓人混淆,大致的區別如下:

重載(overload):
必須在一個域中,函數名稱相同但是函數參數不同,重載的作用就是同一個函數有不同的行爲,因此不是在一個域中的函數是無法構成重載的,這個是重載的重要特徵

覆蓋(override):
覆蓋指的是派生類的虛擬函數覆蓋了基類的同名且參數相同的函數,既然是和虛擬函數掛鉤,說明了這個是一個多態支持的特性,所謂的覆蓋指的是用基類對象的指針或者引用時訪問虛擬函數的時候會根據實際的類型決定所調用的函數,因此此時派生類的成員函數可以"覆蓋"掉基類的成員函數.
注意唯有同名且參數相同還有帶有virtual關鍵字並且分別在派生類和基類的函數才能構成虛擬函數,這個也是派生類的重要特徵.
而且,由於是和多態掛鉤的,所以只有在使用類對象指針或者引用的時候才能使用上.
總之一句話:覆蓋函數都是虛函數,反之不然~~

隱藏(hide):
指的是派生類的成員函數隱藏了基類函數的成員函數.隱藏一詞可以這麼理解:在調用一個類的成員函數的時候,編譯器會沿着類的繼承鏈逐級的向上查找函數的定義,如果找到了那麼就停止查找了,所以如果一個派生類和一個基類都有同一個同名(暫且不論參數是否相同)的函數,而編譯器最終選擇了在派生類中的函數,那麼我們就說這個派生類的成員函數"隱藏"了基類的成員函數,也就是說它阻止了編譯器繼續向上查找函數的定義....
回到隱藏的定義中,前面已經說了有virtual關鍵字並且分別位於派生類和基類的同名,同參數函數構成覆蓋的關係,因此隱藏的關係只有如下的可能:
1)必須分別位於派生類和基類中
2)必須同名
3)參數不同的時候本身已經不構成覆蓋關係了,所以此時是否是virtual函數已經不重要了
  當參數相同的時候就要看時候有virtual關鍵字了,有的話就是覆蓋關係,沒有的時候就是隱藏關係了

上面的解說大體把三者的區別給說清楚了,但是還有一些疑惑的地方,以下以代碼例子說明.

很多人分辨不清隱藏和覆蓋的區別,因爲他們都是發生在基類和派生類之中的.但是它們之間最爲重要的區別就是:
覆蓋的函數是多態的,是存在於vtbl之中的函數才能構成"覆蓋"的關係,而隱藏的函數都是一般的函數,不支持多態,在編譯階段就已經確定下來了.

 

class  Base
  {
 public :
 virtual   void  f( float  x)  {cout << " Base::f(folat) " << x << endl;} 
          void  g( float  x)  {cout << " Base::g(float) " << x << endl;}    
} ;

 class  Derived: public  Base
  {
 public  :
     virtual   void  f( float  x)  {cout << " Derived::f(float) " << x << endl;} 
              void  g( int  x)  {cout << " Deriver::g(int) " << x << endl;} 
} ;

 int  main()
  {
    Derived d;
    Base  * pb =& d;
    Derived  * pd =& d;
    pb -> f( 3.14f );
    pd -> f( 3.14f );
    pb -> g( 3.14f );    // 輸出結果:Base::g(float)3.14 
     pd -> g( 3.14f );    // 輸出結果:Dervied::g(int)3 
      return   0 ;
} 


在調用f函數的時候,派生類Derived的f函數覆蓋了基類Base的f函數,而派生類Derived的g函數隱藏了基類Base的g函數.
爲什麼?理由很簡單,f函數是virtual函數,但是g函數不是.我們可以把Base類和Derived類看成這樣的一個struct:

struct Base
{
    void          (*g)(float);  // Base類型的函數指針,不可變
    struct VTABLE  *__vptr;     // 虛擬函數指針數組,可變
};

void __Baseg(float)
{
    cout<<"Base::g(folat)"<<x<<endl;
}

struct Derived
{
    void          (*g)(float);  // Derived類型的函數指針,不可變
    struct VTABLE  *__vptr;     // 虛擬函數指針數組,可變
};

void __Derivedg(float)
{
    cout<<"Deriver::g(int)"<<x<<endl;
}

struct VTABLE
{
    void          (*f)(float);  // 函數指針
};

void __Basef(float)
{
    cout<<"Base::f(folat)"<<x<<endl;
}

void __Derivedf(float)
{
    cout<<"Deriver::f(int)"<<x<<endl;
}


在程序編譯的時候,函數指針f就已經是確定的了,但是__vptr根據不同的而有分別,而這個變化是運行期動態決定的.
也就是說:f的地址不可變,__vptr可變.
回到上面的例子中,Base *pb=&d;的時候只是用Derived類對象d的__vptr修改了Base類pb的__vptr指針,但是當Base類成員建立的
時候f函數指針就是不能改變的.

當函數被聲明爲virtual的時候,就激活了多態機制,程序在運行的時候會根據類型的實際類型到VTABLE中查找函數指針,因此對函數g的調用就是這樣子的:
pb->__vptr->g();
而對f的調用就是一般的類成員函數指針的調用了:pb->f(),因爲這個類型在程序編譯的時候已經確認了,所以在程序運行的時候是不能發生改變的.

綜上,可以把
Derived d;
Base *pb=&d;
的過程分解爲:
d.g = __Derivedg;
d.__vptr->f = __Derivedf;
pb->g = __Baseg;            // 這裏根據指針的真正類型確定函數指針
pb->__vptr = d.__vptr;      // 這裏只是簡單的指針賦值,因此訪問到的就是Derived的函數了
最後在調用:
pb->f(3.14f);
pb->g(3.14f);
實際上是:
pb->__vptr->__Derivedf(3.14f);
__Baseg(3.14f);
這麼寫就明白最後在調用的時候爲什麼會用那樣的結果了,可以看出多了一個__vptr這個間接層實現了所謂的"動態綁定".

最後,需要說明的一點是:實際上在c++中,非static和非virtual的函數指針並不會在一個class中保存它的函數指針,上面把函數g的指針寫在struct裏面只是爲了方便說明這樣的問題:在編譯階段這個函數就已經是確定的不可改變的了.特此說明一下.


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章