免费高清特黄a大片,九一h片在线免费看,a免费国产一级特黄aa大,国产精品国产主播在线观看,成人精品一区久久久久,一级特黄aa大片,俄罗斯无遮挡一级毛片

分享

c++繼承關(guān)系中成員函數(shù)的重載、重寫、重定義之間的區(qū)別

 頭號碼甲 2021-04-26

1、Override、Overload、Redefine

  • Overload
    重載只能發(fā)生在類內(nèi)部,不能發(fā)生在子類和父類的繼承中。具體來說,如果子類中有父類同名、同返回值類型,但是不同參數(shù)列表,這兩個在不同類的函數(shù)是不能發(fā)生重載的。

  • Override
    重寫即就是子類將父類中的方法進行改寫。在實例化Parent *p = new Son(),即創(chuàng)建指針類型為父類,指向子類空間的指針,能看到重寫和重定義的區(qū)別。

  • Redefine
    重定義亦是發(fā)生在在繼承的過程中,這個和重寫容易發(fā)生混淆。主要區(qū)別根據(jù)父類中被重寫或重定義的成員函數(shù)有無virtual關(guān)鍵字來討論。如果沒有virtual關(guān)鍵字,只要函數(shù)名相同,都會發(fā)生函數(shù)的重定義,或者說隱藏,即子類成員函數(shù)隱藏父類同名的成員函數(shù);如果有virtual關(guān)鍵字,首先要保證返回值類型要相同(個人在測試中發(fā)現(xiàn),在子類中,只有將保持返回值類型、函數(shù)名相同,才能進行下一步的重寫或重定義),再判斷是發(fā)生重載還是重定義,如果參數(shù)列表相同,則發(fā)生重寫,如果不相同,則是重定義。


2、三者之間的區(qū)別

  • 重載 overload

    • 發(fā)生在相同的作用域(子類和父類不在同一個作用域)
    • 函數(shù)名要相同
    • 參數(shù)列表不同,包括參數(shù)類型、參數(shù)個數(shù)、參數(shù)的順序
    • 有無virtual關(guān)鍵字都可以發(fā)生
    • 返回值可以不同
  • 重寫 override

    • 不同的作用域(兩個同名函數(shù)分別在父類和子類)
    • 相同的函數(shù)名
    • 相同的參數(shù)列表
    • 被重寫父類中的成員函數(shù)必須有關(guān)鍵字'virtual'
    • 相同的返回值類型
    • 被重寫的成員函數(shù)訪問權(quán)限可以被修改,public、protect 或者其他。
  • 重定義 redefine

    • 不同的作用域
    • 函數(shù)名相同
    • 返回值類型可以不同(沒有關(guān)鍵字virtual的情況),但是如果有virtual關(guān)鍵字,必須保證返回類型相同,否則編譯報錯。
    • 父類函數(shù)沒有關(guān)鍵字virtual,參數(shù)列表可同可不同;父類函數(shù)有關(guān)鍵字virtual,參數(shù)列表必須不同。

舉個例子說明一下:

class Base{
public:
    int param3 = 0;
    void func1(){cout<<"This is Base::func1()"<<endl;}
    void func2(int a){cout<<"This is Base::func2(int a)"<<endl;}
    void func2(char c){cout<<"This is Base::func2(char c)"<<endl;}
    void func3(){cout<<"This is Base::func3()"<<endl;}
    virtual void func4(){cout<<"This is Base::func4()"<<endl;}
    virtual void func5(){cout<<"This is Base::func5()"<<endl;}
    virtual int func6(){cout<<"This is int Base::func6()"<<endl;}
};

class Son: public Base
{
public:
    int param = 1;
    int func1(){cout<<"This is Son::func1()"<<endl;}
    void func2(double e){cout<<"This is Son::func2()"<<endl;}
    void func3(){cout<<"This is Son::func3()"<<endl;}
    void func4(){cout<<"This is Son::func4()"<<endl;}
    void func5(int a){cout<<"This is Son::func5(int a)"<<endl;}
//    double func6(){cout<<"This is Son::func6()"<<endl;}
};

int main() {
    Son s;
    Base b;
    Base *bp = new Son();
    s.func1();
    s.func2(1.1);
    s.func2('c');
    s.func4();
    s.func5(2);
    cout<<"--------------------------------"<<endl;
    b.func1();
    b.func2(1);
    b.func2('c');
    b.func3();
    cout<<"--------------------------------"<<endl;
    bp->func1();
    bp->func2(1);
    bp->func2('e');
    bp->func3();
    bp->func4();
    bp->func5();
    return 0;
}

輸出如下:

This is Son::func1()
This is Son::func2()
This is Son::func2()
This is Son::func4()
This is Son::func5(int a)
--------------------------------
This is Base::func1()
This is Base::func2(int a)
This is Base::func2(char c)
This is Base::func3()
--------------------------------
This is Base::func1()
This is Base::func2(int a)
This is Base::func2(char c)
This is Base::func3()
This is Son::func4()
This is Base::func5()

分別創(chuàng)建子類、父類、指針類型為父類指向子類空間的指針。(1)父類中的func2發(fā)生重載,主要在父類內(nèi)部產(chǎn)生(應(yīng)該說相同作用域),因為給s.func2('c')傳入字符的時候,只會調(diào)用子類函數(shù),不會調(diào)用父類的func2(char c)。而子類中的func2對父類的func2發(fā)生了重定義,并對其做了隱藏,所以調(diào)用的時候才會調(diào)用到子類的func2(double )。從s.func1()、s.func2()、s.func3()都發(fā)生了重定義,所以在繼承的過程中,如果沒有virtual關(guān)鍵字,只要函數(shù)名相同,不管參數(shù)類型、返回值類型,都會發(fā)生重定義。(2)針對有virtual關(guān)鍵字的情況,在函數(shù)名相同的情況下,首先要保證返回值類型相同,否則編譯不過,如果參數(shù)列表相同,則發(fā)生重寫,不同則發(fā)生重定義,例如bp->func4()bp->func5(),這里bp調(diào)用函數(shù)的處理取決于是否重寫的函數(shù)(虛函數(shù)的特性)。


參考文獻

    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點。請注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購買等信息,謹防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點擊一鍵舉報。
    轉(zhuǎn)藏 分享 獻花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多