Overload
重載只能發(fā)生在類內部,不能發(fā)生在子類和父類的繼承中。具體來說,如果子類中有父類同名、同返回值類型,但是不同參數(shù)列表,這兩個在不同類的函數(shù)是不能發(fā)生重載的。
Override
重寫即就是子類將父類中的方法進行改寫。在實例化Parent *p = new Son()
,即創(chuàng)建指針類型為父類,指向子類空間的指針,能看到重寫和重定義的區(qū)別。
Redefine
重定義亦是發(fā)生在在繼承的過程中,這個和重寫容易發(fā)生混淆。主要區(qū)別根據(jù)父類中被重寫或重定義的成員函數(shù)有無virtual
關鍵字來討論。如果沒有virtual
關鍵字,只要函數(shù)名相同,都會發(fā)生函數(shù)的重定義,或者說隱藏,即子類成員函數(shù)隱藏父類同名的成員函數(shù);如果有virtual
關鍵字,首先要保證返回值類型要相同(個人在測試中發(fā)現(xiàn),在子類中,只有將保持返回值類型、函數(shù)名相同,才能進行下一步的重寫或重定義),再判斷是發(fā)生重載還是重定義,如果參數(shù)列表相同,則發(fā)生重寫,如果不相同,則是重定義。
重載 overload
virtual
關鍵字都可以發(fā)生重寫 override
public
、protect
或者其他。重定義 redefine
virtual
的情況),但是如果有virtual
關鍵字,必須保證返回類型相同,否則編譯報錯。virtual
,參數(shù)列表可同可不同;父類函數(shù)有關鍵字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ā)生重載,主要在父類內部產生(應該說相同作用域),因為給s.func2('c')
傳入字符的時候,只會調用子類函數(shù),不會調用父類的func2(char c)
。而子類中的func2
對父類的func2
發(fā)生了重定義,并對其做了隱藏,所以調用的時候才會調用到子類的func2(double )
。從s.func1()
、s.func2()
、s.func3()
都發(fā)生了重定義,所以在繼承的過程中,如果沒有virtual
關鍵字,只要函數(shù)名相同,不管參數(shù)類型、返回值類型,都會發(fā)生重定義。(2)針對有virtual
關鍵字的情況,在函數(shù)名相同的情況下,首先要保證返回值類型相同,否則編譯不過,如果參數(shù)列表相同,則發(fā)生重寫,不同則發(fā)生重定義,例如bp->func4()
和bp->func5()
,這里bp調用函數(shù)的處理取決于是否重寫的函數(shù)(虛函數(shù)的特性)。