C++虛函數(shù)


title: 理解C++虛函數(shù)
date: 2018-11-11 15:31:26


1. 簡單介紹

C++虛函數(shù)是定義在基類中的函數(shù),子類必須對其進(jìn)行覆蓋。在類中聲明(無函數(shù)體的形式叫做聲明)虛函數(shù)的格式如下:

virtual void display();

2. 虛函數(shù)的作用

虛函數(shù)有兩大作用:

(1)定義子類對象,并調(diào)用對象中未被子類覆蓋的基類函數(shù)A。同時在該函數(shù)A中,又調(diào)用了已被子類覆蓋的基類函數(shù)B。那此時將會調(diào)用基類中的函數(shù)B,可我們本應(yīng)該調(diào)用的是子類中的覆蓋函數(shù)B。虛函數(shù)即能解決這個問題。

以下是沒有使用虛函數(shù)的例子:

#include<iostream>

using namespace std;

// 基類 Father
class Father {
public:
    void display() {
        cout<<"Father::display()\n";
    }
    // 在函數(shù)中調(diào)用了,子類覆蓋基類的函數(shù)display()
    void fatherShowDisplay() {
        display();
    }
};

// 子類Son
class Son:public Father {
public:
    // 重寫基類中的display()函數(shù)
    void display() {
        cout<<"Son::display()\n";
    }
};

int main() {
    Son son;                    // 子類對象
    son.fatherShowDisplay();    // 通過基類中未被覆蓋的函數(shù),想調(diào)用子類中覆蓋的display函數(shù)
}

該例子的運行結(jié)果是: Father::display()

以下是使用虛函數(shù)的例子:

#include<iostream>

using namespace std;

// 基類 Father
class Father {
public:
    virtual void display() {
        cout<<"Father::display()\n";
    }

    // 在函數(shù)中調(diào)用了,子類覆蓋基類的函數(shù)display()
    void fatherShowDisplay() {
        display();
    }
};

// 子類Son
class Son:public Father {
public:
    // 重寫基類中的display()函數(shù)
    void display() {
        cout<<"Son::display()\n";
    }
};

int main() {
    Son son;                    // 子類對象
    son.fatherShowDisplay();    // 通過基類中未被覆蓋的函數(shù),想調(diào)用子類中覆蓋的display函數(shù)
}

該例子的運行結(jié)果是: Son::display()

(2)在使用指向子類對象的基類指針,并調(diào)用子類中的覆蓋函數(shù)時,如果該函數(shù)不是虛函數(shù),那么將調(diào)用基類中的該函數(shù);如果該函數(shù)是虛函數(shù),則會調(diào)用子類中的該函數(shù)。

以下是沒有使用虛函數(shù)的例子:

#include<iostream>

using namespace std;

// 基類 Father
class Father {
public:
    void display() {
        cout<<"Father::display()\n";
    }
};

// 子類Son
class Son:public Father {
public:
    // 覆蓋基類中的display函數(shù)
    void display() {
        cout<<"Son::display()\n";
    }
};

int main() {
    Father *fp;     // 定義基類指針
    Son son;        // 子類對象
    fp=&son;        // 使基類指針指向子類對象
    fp->display();  // 通過基類指針想調(diào)用子類中覆蓋的display函數(shù)
}

該例子的運行結(jié)果是: Father::display()
結(jié)果說明,通過指向子類對象的基類指針調(diào)用子類中的覆蓋函數(shù)是不能實現(xiàn)的,因此虛函數(shù)應(yīng)運而生。

以下是使用虛函數(shù)的例子:

#include<iostream>

using namespace std;

// 基類 Father
class Father {
public:
    // 定義了虛函數(shù)
    void virtual display() {
        cout<<"Father::display()\n";
    }
};

// 子類Son
class Son:public Father {
public:
    // 覆蓋基類中的display函數(shù)
    void display() {
        cout<<"Son::display()\n";
    }
};

int main() {
    Father *fp;     // 定義基類指針
    Son son;        // 子類對象
    fp=&son;        // 使基類指針指向子類對象
    fp->display();  // 通過基類指針想調(diào)用子類中覆蓋的display函數(shù)
}

該例子的運行結(jié)果是: Son::display()

3. 虛函數(shù)的實際意義

或許,很多小伙伴都會有這樣一個疑問:如果想調(diào)用子類中的覆蓋函數(shù),直接通過子類對象,或者指向子類對象的子類指針來調(diào)用,不就沒這個煩惱了嗎?要虛函數(shù)還有什么用呢?

其實不然,虛函數(shù)的實際意義非常之大。比如在實際開發(fā)過程中,會用到別人封裝好的框架和類庫,我們可以通過繼承其中的類,并覆蓋基類中的函數(shù),來實現(xiàn)自定義的功能。

但是,有些函數(shù)是需要框架來調(diào)用,并且API需要傳入基類指針類型的參數(shù)。而使用虛函數(shù)就可以,將指向子類對象的基類指針來作為參數(shù)傳入API,讓API能夠通過基類指針,來調(diào)用我們自定義的子類函數(shù)。這就是多態(tài)性的真正體現(xiàn)。

4. 淺談虛函數(shù)的原理

參考:C++中的虛函數(shù)(表)實現(xiàn)機制以及用C語言對其進(jìn)行的模擬實現(xiàn)

虛函數(shù)的本質(zhì)是一個簡單的虛函數(shù)表

當(dāng)一個類存在虛函數(shù)時,通過該類創(chuàng)建的對象實例,會在內(nèi)存空間的前4字節(jié)保存一個指向虛函數(shù)表的指針__vfptr

__vfptr指向的虛函數(shù)表,是類獨有的,而且被該類的所有對象共享。虛函數(shù)表的實質(zhì),是一個虛函數(shù)地址的數(shù)組,它包含了類中每個虛函數(shù)的地址,既有當(dāng)前類定義的虛函數(shù),也有覆蓋父類的虛函數(shù),也有繼承而來的虛函數(shù)。

當(dāng)子類覆蓋了父類的虛函數(shù)時,子類虛函數(shù)表將包含子類虛函數(shù)的地址,而不會有父類虛函數(shù)的地址。

同時,當(dāng)用基類指針指向子類對象時,基類指針指向的內(nèi)存空間中的__vfptr依舊指向了子類的虛函數(shù)表。所以,基類指針依舊會調(diào)用子類的虛函數(shù)。

見如下示例:

4.1. 自己定義了虛函數(shù)的類

class Base1 {
public:
    int base1_1;
    int base1_2;

    virtual void base1_fun1() {}
    virtual void base1_fun2() {}
};

定義兩個對象:

Base1 b1;
Base1 b2;

兩個對象的內(nèi)存空間分配如下:

1.png

4.2. 既包含覆蓋虛函數(shù),又包含繼承虛函數(shù)的類

class Base1 {
public:
    int base1_1;
    int base1_2;

    virtual void base1_fun1() {}
    virtual void base1_fun2() {}
};

class Derive1 : public Base1 {
public:
    int derive1_1;
    int derive1_2;

    // 覆蓋基類函數(shù)
    virtual void base1_fun1() {}
};

定義一個子類對象:

Derive1 d1;

其內(nèi)存空間如下:

2.png

由圖可以看出:

Base1 *b_p = &d1;     // 指向子類對象的基類指針
b_p->base1_fun1();    // 調(diào)用子類虛函數(shù)
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,333評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,491評論 3 416
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,263評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,946評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 71,708評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,186評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,255評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,409評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,939評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,774評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,976評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,518評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,209評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,641評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,872評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,650評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 47,958評論 2 373

推薦閱讀更多精彩內(nèi)容