一文搞懂C++多態(tài)的用法
前言
C++多態(tài)是在繼承的基礎上實現(xiàn)的,了解多態(tài)之前我們需要掌握一定的C++繼承的知識,本文將介紹C++中多態(tài)的概念,構成條件以及用法。
1.多態(tài)的概念
多態(tài),通俗來講就是多種形態(tài),具體點就是去完成某個行為,當不同的對象去完成時會產(chǎn)生出不同的狀態(tài)。比如,在買票這一行為,普通人買票是全價買票,學生買票是半價買票,而軍人買票是優(yōu)先買票;再比如動物園的動物叫這個行為,不同的動物叫聲是不一樣的。這些都是生活中多態(tài)的例子。
2.C++中多態(tài)的分類
(1)靜態(tài)多態(tài)
靜態(tài)多態(tài)是指在編譯時實現(xiàn)的多態(tài),比如函數(shù)重載,看似是調(diào)用同一個函數(shù)其實是在調(diào)用不同的。
比如我們使用cout這個對象來調(diào)用<<時:
int i=1; double d=2.2; cout<<i<<endl; cout<<d<<endl;
雖然調(diào)用的都是<<,但其實調(diào)用的是不同的操作符重載之后的函數(shù)。
函數(shù)重載在之前的文章中詳細講解過,這里就不再贅述。
(2)動態(tài)多態(tài)
動態(tài)多態(tài)也就是我們通常所說的多態(tài),本文以下內(nèi)容均為動態(tài)多態(tài)內(nèi)容。動態(tài)多態(tài)是在運行中實現(xiàn)的,
當一個父類對象的引用或者指針接收不同的對象(父類對象or子類對象)后,調(diào)用相同的函數(shù)會調(diào)用不同的函數(shù)。
這段話也許比較繞,這里只是給出一個概念,可以結(jié)合下面的例子來進行理解。
3.多態(tài)的構成條件
(1)舉例
我們先根據(jù)一個構成例子來理解多態(tài)構成的條件:
#include<iostream> #include<string> using namespace std; class Person { public: virtual void BuyTicket() { cout << "全價買票" << endl; } }; class Student :public Person { public: virtual void BuyTicket() { cout << "半價買票" << endl; } }; void Func(Person& p) { p.BuyTicket(); } int main() { Person p1; Student p2; Func(p1); Func(p2); }
我們先來看這樣一段代碼,其中子類Student繼承了父類。運行起來打印的結(jié)果是:
我們在反觀上述中動態(tài)多態(tài)的定義,用父類的引用或者指針(這里使用的是Person& p)來接收不同類型的對象(p1和p2),該引用或指針調(diào)用相同的函數(shù)(都調(diào)用了p.BuyTicket()),都調(diào)用了各自類中不同的函數(shù)(打印的結(jié)果不同)。我們將這一過程稱為動態(tài)多態(tài)。
如果我們不傳指針或者引用,那么將不構成多態(tài)(原理會在多態(tài)原理中詳細解讀)。
(2)兩個概念
在解釋多態(tài)的構成條件之前我們還需要了解兩個概念。
虛函數(shù)
虛函數(shù),即被virtual修飾的類成員函數(shù)稱為虛函數(shù)。
比如上面代碼中父類和子類的成員函數(shù)就是虛函數(shù)。
virtual void BuyTicket() { cout << "全價買票" << endl; }
關于虛函數(shù)還需要注意幾點:
1.普通的函數(shù)不能是虛函數(shù),只能是類中的函數(shù)。
2.靜態(tài)成員函數(shù)不能加virtual
總結(jié)起來就是只能是類的非靜態(tài)成員函數(shù)才能去形成虛函數(shù)。
虛函數(shù)的重寫
虛函數(shù)的重寫又稱為虛函數(shù)的覆蓋(重寫是表面意義上的,覆蓋是指原理上的):派生類中有一個根基類完全相同的虛函數(shù)(即派生類虛函數(shù)與基類虛函數(shù)的返回值類型,函數(shù)名,參數(shù)列表完全相同),稱子類的虛函數(shù)重寫了父類的虛函數(shù)。
//父類中的虛函數(shù) virtual void BuyTicket() { cout << "全價買票" << endl; } //子類中的虛函數(shù) virtual void BuyTicket() { cout << "半價買票" << endl; }
兩個虛函數(shù)滿足返回值類型相同,函數(shù)名相同,參數(shù)列表相同。因此子類的虛函數(shù)重寫了父類的虛函數(shù)。
注意,只有虛函數(shù)才能構成重寫。
(3)多態(tài)的構成條件
多態(tài)的構成滿足兩個條件:
1.必須通過基類的指針或者引用調(diào)用虛函數(shù)。
2.被調(diào)用的虛函數(shù)的派生類必須完成了對基類虛函數(shù)的重寫。
我們在來看上面的代碼,確實滿足該條件:
1.使用了父類引用p來調(diào)用虛函數(shù)。
2.派生類的虛函數(shù)完成了對基類的虛函數(shù)的重寫。
我們首先要明確使用多態(tài)的目的,就是使用不同的對象去完成同一個任務的時候會產(chǎn)生不同的結(jié)果。
如果我們拿掉以上任何一個條件都不會再構成多態(tài),比如我們不使用指針或者引用去接收對象從而調(diào)用虛函數(shù),而是使用對象呢?
void Func(Person p) { p.BuyTicket(); }
此時我們會發(fā)現(xiàn),打印的結(jié)果發(fā)生了變化:
這是不滿足我們的預期的,因為不同的對象傳給了p,p調(diào)用相同的函數(shù)卻打印了相同的結(jié)果。
我們還可以將更改參數(shù)列表或者將父類的virtual拿掉,發(fā)現(xiàn)依然不是我們想要的結(jié)果。
但是有兩個特殊的情況除外:
4.虛函數(shù)重寫的兩個例外
(1)協(xié)變
如果我們將父類和子類中的虛函數(shù)的返回值設為不同,可能會發(fā)生如下報錯:
協(xié)變指的是:派生類重寫基類虛函數(shù)時,與基類虛函數(shù)返回值類型不同。即基類的虛函數(shù)返回基類對象的指針或者引用,派生類虛函數(shù)返回派生類對象的指針或者引用時,稱為協(xié)變。
簡單來說就是兩者的返回值是父子關系的指針或者引用。
舉一個例子:
class A{}; class B:public A {}; class Person { public: virtual A* BuyTicket() { A a; cout << "全價買票" << endl; return &a; } }; class Student :public Person { public: virtual B* BuyTicket() { B b; cout << "半價買票" << endl; return &b; } };
我們將上一段代碼進行了改寫,定義了B繼承A,而在Person和Student兩個類中的虛函數(shù)中將返回值分別置為A和B,由于A和B是繼承關系,所以仍然可以構成多態(tài),我們稱派生類的虛函數(shù)為基類的虛函數(shù)的協(xié)變。
注意返回值必須是指針或者引用,對象不會構成協(xié)變。
(2)析構函數(shù)的重寫
首先我們先回顧一下沒有構成多態(tài)的析構函數(shù)調(diào)用:只需要子類對象銷毀時無需手動銷毀父類對象,會自動調(diào)用父類對象的析構函數(shù)。
1.如果基類的析構函數(shù)為虛函數(shù),此時子類的析構函數(shù)無論加不加virtual,都是對父類的析構函數(shù)的重寫。
2.雖然子類和父類的析構函數(shù)的函數(shù)名不同,但其實編譯器對析構函數(shù)的名稱進行了特殊的處理,都處理成了destructor。
下面舉例說明,將Person和Student寫入析構函數(shù):
//父類中的析構函數(shù) virtual ~Person() { cout << "~Person" << endl; } //子類中的析構函數(shù) virtual ~Student() { cout << " ~Student" << endl; } //主函數(shù) Person* p1 = new Person; Person* p2 = new Student; delete p1; delete p2;
構成多態(tài)的結(jié)果是,Person*類型的p1和p2,接收兩個不同類型的對象即Person類型和Student類型,在調(diào)用析構函數(shù)的時候可以分開調(diào)用(子類對象調(diào)用子類的析構函數(shù),父類對象調(diào)用父類的析構函數(shù)。)
我們將上述代碼運行一下,會發(fā)現(xiàn):
結(jié)果的確是如此,當析構父類對象時,調(diào)用父類的析構函數(shù),當析構子類對象時,調(diào)用的是子類的析構函數(shù)和父類的析構函數(shù)。
如果我們不使用父類指針進行管理,而是使用對象來接收子類對象呢?
Student p2; Person p3 = p2;
此時我們發(fā)現(xiàn)打印的結(jié)果是:
在析構p3的時候,并沒有根據(jù)按Student類的規(guī)則來進行析構。
同時,當我們將派生類的virtual去掉的時候,仍然可以構成多態(tài),這與底層原理有關,在下面的介紹中會提及。為了統(tǒng)一性,不建議將virtual拿掉,C++大佬為了防止發(fā)生不必要的內(nèi)存泄漏,所以設置了這一規(guī)則。這就導致所有的其實派生類的所有虛函數(shù)virtual都可以省略。這是由于其繼承了基類的virtual屬性,具體的還要在底層去理解,再強調(diào)一遍,盡量不要在派生類中省略virtual。
5.final與override
(1)final
限制類不被繼承
但我們想要設計一個不被繼承的類時,目前我們知道的有一種方法:就是將父類的構造函數(shù)設為私有(這是因為子類需要調(diào)用父類的構造函數(shù)來進行初始化)。如果使用這種方式,定義父類對象的話需要使用單例模式。
final提供了另一種方式來限制一個類不會被繼承。
只需要在不想被繼承的類后加final即可:
class Person final { public: virtual A* BuyTicket() { A a; cout << "全價買票" << endl; return &a; } virtual ~Person() { cout << "~Person" << endl; } };
此時如果子類去繼承Person的話會報錯。
限制虛函數(shù)不被重寫
當我們在函數(shù)后加上final的時候,該虛函數(shù)將不能被子類中的虛函數(shù)重寫,否則會發(fā)生報錯。
virtual A* BuyTicket() final { A a; cout << "全價買票" << endl; return &a; }
(2)override
將override放在子類的重寫的虛函數(shù)后,判斷是否完成重寫(重寫的是否正確)
virtual B* BuyTicket(int i=10) override { B b; cout << "半價買票" << endl; return &b; }
注意:final關鍵字放在父類的位置,override關鍵字放在子類的位置。
6.抽象類
在虛函數(shù)的后面加上=0,則這個函數(shù)為純虛函數(shù)。包含純虛函數(shù)的類叫做抽象類(也叫做接口類),抽象類不能實例化出對象。派生類繼承后也不能實例化出對象。**只有重寫虛函數(shù),派生類才能實例化出對象。**注意雖然不能實例化出對象,但是可以定義指針。
抽象類的存在本質(zhì)上來說就是希望我們在派生類中重寫父類的虛函數(shù)。抽象類中的虛函數(shù)一般只聲明,不實現(xiàn),因為沒有意義。我們可以搭配override來使用。
//將父類中寫入純虛函數(shù),父類變成抽象類 class Person { public: virtual A* BuyTicket() =0//純虛函數(shù) { A a; cout << "全價買票" << endl; return &a; } virtual ~Person() { cout << "~Person" << endl; } };
此時子類必須只有重寫虛函數(shù)才能定義對象。通常情況下現(xiàn)實中沒有的事物,定義成抽象類會比較合適。
雖然我們不能使用抽象類來定義對象,但是我們可以使用抽象類來定義指針。
class Car { public: virtual void Drive() = 0 { cout << " Car" << endl; } }; class Benz :public Car { public: virtual void Drive() { cout << "Benz" << endl; } void f() { cout << "f()" << endl; } }; int main() { //Car* p = nullptr; //p->Drive();//程序會崩潰 Car* a = new Benz; a->Drive(); }
我們可以使用父類指針去接收子類對象,同時調(diào)用函數(shù)。但是不能使用父類去創(chuàng)建對象。
7.總結(jié)
C++多態(tài)的目的在于當我們使用父類的指針或者引用去接收子類的對象后,接收不同的子類對象的父類指針或者引用調(diào)用的相同的函數(shù)產(chǎn)生的結(jié)果不同。
重點在于實現(xiàn)多態(tài)的幾個條件:
一是用父類的指針或者引用來接收。
二是子類必須對父類的虛函數(shù)進行重寫。
到此這篇關于一文搞懂C++多態(tài)的用法的文章就介紹到這了,更多相關C++ 多態(tài)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
C/C++實現(xiàn)經(jīng)典象棋游戲的示例代碼
中國象棋是起源于中國的一種棋,屬于二人對抗性游戲的一種,在中國有著悠久的歷史。本文將利用C++實現(xiàn)這一經(jīng)典游戲,快跟隨小編一起學習一下吧2022-06-06C語言通過gets和gets_s分別實現(xiàn)讀取含空格的字符串
在遇到包含空格的字符串輸入時該如何讀取呢?如果使用scanf以%s格式去讀取輸入的字符串,遇到空格就讀取結(jié)束了,顯然這樣是讀取不了的。本文就將介紹兩個可以對含空格字符串讀取的庫函數(shù)------gets和gets_s函數(shù),感興趣的可以了解一下2021-12-12深入探討:main函數(shù)執(zhí)行完畢后,是否可能會再執(zhí)行一段代碼?
本篇文章是對main函數(shù)執(zhí)行完畢后,是否可能會再執(zhí)行一段代碼,進行了詳細的分析介紹,需要的朋友參考下2013-05-05淺談Windows系統(tǒng)下C語言編程中Glib庫的使用
這篇文章主要介紹了Windows系統(tǒng)下C語言編程中Glib庫的使用,Glib庫在多線程編程中經(jīng)常可以用到,需要的朋友可以參考下2016-02-02