重寫
- 重寫的定義:重寫發生在基類和派生類的繼承關系之中,被定義為虛函數的基類成員函數,由派生類進行重新定義和實現,同時隱藏掉基類的方法(即派生類調用該重寫方法時,會使用派生類重定義的方法,而非基類方法)。例如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
#include <iostream> using std::cout; using std::endl; class Base { public : Base(){}; ~Base(){}; virtual void fun() {cout << "Base class" << endl;} }; class Derived : public Base { public : Derived(){}; ~Derived(){}; void fun() override {cout << "Derived class" << endl;}; }; int main() { Derived DerivedClass; Base BaseClass; DerivedClass.fun(); BaseClass.fun(); } |
輸出為:
Derived class
Base class
重寫的注意點:
- 重寫時父類需要將成員函數加上virtual關鍵字,子類在重寫的時候需要保證返回類型,參數個數,參數類型一致
- 重寫的成員函數訪問修飾符可變,即父類在private中聲明的虛函數,子類可以重寫為public
- 可以使用協變返回類型進行虛函數的重寫,將返回子類重寫時的會返回來類型
協變返回類型:在C++中,只要原來的返回類型是指向類的指針或引用,新的返回類型是指向派生類的指針或引用,覆蓋的方法就可以改變返回類型。這樣的類型稱為協變返回類型(Covariant returns type).
通俗的來講,原本重寫需要保證虛函數的返回類型相同,但是如果返回的類型時指針或者是引用,在保證該指針或者引用是具有繼承關系的情況下,重寫的虛函數可以返回子類的指針或者是引用,例如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
class Base { public : Base(){}; ~Base(){}; virtual void fun() {cout << "Base class" << endl;} virtual Base* fun2() { return this ;}; }; class Derived : public Base { public : Derived(){}; ~Derived(){}; void fun() override {cout << "Derived class" << endl;}; Derived* fun2() override { return this ;}; }; |
派生類重寫了基類的fun2函數,基類返回基類指針,派生類返回派生類指針。
重載
- 重載的定義:重載指同一可訪問區內(代碼塊內)被聲明的幾個具有不同參數列表也即函數簽名(參數的類型,個數,順序不同)的同名函數,根據參數列表確定調用哪個函數,重載不關心函數返回類型。例如:
1
2
3
4
5
|
int test(); int test( int a); int test( int a, double b); int test( double a, int a); int test(string s); |
需要注意的點:
- 重載只和函數簽名有關,和函數的返回類型無關
- 重載發生在統一作用域(代碼塊)中
- 類中靜態函數可以和普通成員函數進行重載
- 重載多用于運算符
隱藏
- 隱藏的定義: 指不同作用域中定義的同名函數構成隱藏(不要求函數返回值和函數參數類型相同)。比如派生類成員函數隱藏與其同名的基類成員函數、類成員函數隱藏全局外部函數。
隱藏比較簡單粗暴,只要滿足在不同的作用域中,且名稱相同即可發生隱藏,例如類中成員函數隱藏全局函數,派生類的成員函數隱藏基類成員函數。重寫是一種特殊的隱藏,重寫是動態多態的一種體現,會影響到虛表,虛指針等編譯和運行時行為。
需要注意的點:
- 在函數查找時,名字查找先于類型檢查
- 只要滿足同名函數就可能會發生隱藏
總結
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關注服務器之家的更多內容!
原文鏈接:https://blog.csdn.net/qq_41719595/article/details/120355819