欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++虛函數(shù)表和虛析構(gòu)介紹

 更新時(shí)間:2021年11月12日 10:39:57   作者:學(xué)渣的C/C++  
這篇文章主要介紹了C++虛函數(shù)表和虛析構(gòu),虛函數(shù)表是C++實(shí)現(xiàn)多態(tài)的基礎(chǔ),多態(tài)是面向?qū)ο蟮娜筇匦灾唬旅嫖恼挛覀円黄饋?lái)看看詳細(xì)內(nèi)容,需要的朋友可以參考一下

1、虛函數(shù)表

虛函數(shù)表是C++實(shí)現(xiàn)多態(tài)的基礎(chǔ),多態(tài)是面向?qū)ο蟮娜筇匦灾?,多態(tài)有利于提高代碼的可讀性,便于后期代碼的擴(kuò)展和維護(hù)。我們都知道多態(tài)的實(shí)現(xiàn)是基于虛函數(shù)表,那么虛函數(shù)表是什么時(shí)候創(chuàng)建的呢?虛函數(shù)表是怎么實(shí)現(xiàn)多態(tài)的功能的呢?

首先應(yīng)該明確多態(tài)也稱為動(dòng)態(tài)多態(tài),他是在程序運(yùn)行時(shí)候確定函數(shù)地址的,也就是程序在運(yùn)行時(shí),如果類(lèi)成員函數(shù)加了virtual關(guān)鍵字,就會(huì)建立一個(gè)虛函數(shù)指針(vfptr)指針指向一個(gè)虛函數(shù)表,這個(gè)虛函數(shù)表就保存了虛函數(shù)的地址,子類(lèi)繼承父類(lèi)也自然繼承了虛函數(shù)指針,當(dāng)子類(lèi)重寫(xiě)父類(lèi)的虛函數(shù)時(shí),虛函數(shù)指針?biāo)赶虻奶摵瘮?shù)表中的虛函數(shù)地址就會(huì)被覆蓋,替換成子類(lèi)的虛函數(shù)地址。也就是通過(guò)父類(lèi)的虛函數(shù)指針找到了子類(lèi)的虛函數(shù)地址,進(jìn)而執(zhí)行這個(gè)函數(shù)。

下面我們通過(guò)代碼進(jìn)行詳細(xì)說(shuō)明:

#include <iostream>
using namespace std;

class Base{
public:
    void func(){
        cout << "Base func" << endl;
    }
};

class Son: public Base{
    void func(){
        cout << "Son func" << endl;
    }
};

void test(Base& base) {
    base.func();
}
int main () {
    Son son;

    cout << "sizeof(Base) = " << sizeof(Base) << endl;
    cout << "sizeof(Son)  = " << sizeof(Son) << endl;   
    test(son);
    system("pause");
    return 0;

}

代碼運(yùn)行結(jié)果為:

 因?yàn)楹瘮?shù)成員不占用類(lèi)的大小,所以對(duì)Base類(lèi)和Son類(lèi)輸出大小,都是一個(gè)字節(jié),這一個(gè)字節(jié)是為了可以實(shí)例化類(lèi),通過(guò)引用基類(lèi)引用派生類(lèi),調(diào)用func函數(shù),函數(shù)調(diào)用了基類(lèi)的func,那么如果我們加上virtual關(guān)鍵字后,就不是這種情況了。

#include <iostream>
using namespace std;

class Base{
public:
    virtual void func(){
        cout << "Base func" << endl;
    }
};

class Son: public Base{
    void func(){
        cout << "Son func" << endl;
    }
};


void test(Base& base) {
    base.func();
}
int main () {
    Son son;

    cout << "sizeof(Base) = " << sizeof(Base) << endl;
    cout << "sizeof(Son)  = " << sizeof(Son) << endl;   
    test(son);
    system("pause");
    return 0;

}

代碼運(yùn)行結(jié)果為:

 可以看到加了virtual關(guān)鍵字后,父類(lèi)和子類(lèi)的大小都變成了四字節(jié),這是因?yàn)樯闪颂摵瘮?shù)指針,指針指向虛函數(shù)表,虛函數(shù)表存儲(chǔ)了虛函數(shù)地址,繼承了父類(lèi)的子類(lèi)重寫(xiě)了虛函數(shù),虛函數(shù)表中的函數(shù)地址被替換,再次調(diào)用虛函數(shù)就是調(diào)用了子類(lèi)的函數(shù)func。

2、虛析構(gòu)

虛析構(gòu)主要是為了解決子類(lèi)中有屬性開(kāi)辟到堆區(qū),父類(lèi)指針調(diào)用函數(shù)時(shí),無(wú)法調(diào)用到子類(lèi)的析構(gòu)代碼,導(dǎo)致子類(lèi)堆區(qū)內(nèi)存無(wú)法釋放。

首先我們看一下子類(lèi)堆區(qū)內(nèi)存開(kāi)辟,通過(guò)父類(lèi)指針來(lái)調(diào)用函數(shù),捕捉他們的構(gòu)造函數(shù)和析構(gòu)函數(shù)看下運(yùn)行結(jié)果:

#include <iostream>
using namespace std;

class Base{
public:
    Base(){
        cout << "Base 的構(gòu)造函數(shù)調(diào)用" << endl;
    }

    ~Base(){
        cout << "Base 的析構(gòu)函數(shù)調(diào)用" << endl;
    }
    virtual void func(){
        cout << "Base func" << endl;
    }
};

class Son: public Base{
public:
    Son(int val):m_val(new int (val)) {
        cout << "Son 的構(gòu)造函數(shù)調(diào)用" << endl;
    }

    ~Son(){
        cout << "Son 的析構(gòu)函數(shù)調(diào)用" << endl;
        if (m_val != NULL) {
            
            delete m_val;
            cout << "Son 析構(gòu)函數(shù)的堆內(nèi)存釋放" << endl;
            m_val = NULL;
        }
    }
    void func(){
        cout << "Son func" << endl;
    }

    void funcTest(){
        cout << "funcTest 函數(shù)調(diào)用" << endl;
    }

    int* m_val = NULL;
};


void test() {
   
    Base *base = new Son(10);
    base->func();
    //base->funcTest(); //無(wú)法調(diào)用,因?yàn)樘摵瘮?shù)表中不能找到這個(gè)函數(shù)的地址
    delete base;
    base = NULL;
}


int main () {
    test();    
    system("pause");
    return 0;

}

代碼運(yùn)行結(jié)果為:

 可以明確,通過(guò)父類(lèi)指針來(lái)調(diào)用函數(shù)的時(shí)候,無(wú)法調(diào)用Son類(lèi)的析構(gòu)函數(shù),在Son類(lèi)在堆區(qū)上申請(qǐng)的內(nèi)存就無(wú)法釋放,造成內(nèi)存泄漏。Son類(lèi)的析構(gòu)函數(shù)不能調(diào)用的主要原因就是在虛函數(shù)表中找不到Son的析構(gòu)函數(shù)地址,解決辦法就是把Base類(lèi)的寫(xiě)成虛析構(gòu)函數(shù)或者純虛析構(gòu)函數(shù),

下面給出Base類(lèi)為純虛析構(gòu)函數(shù)的代碼和運(yùn)行結(jié)果:

#include <iostream>
using namespace std;

class Base{
public:
    Base(){
        cout << "Base 的構(gòu)造函數(shù)調(diào)用" << endl;
    }
    virtual ~Base() = 0;

    virtual void func(){
        cout << "Base func" << endl;
    }
};
    Base :: ~Base(){
        cout << "Base 的析構(gòu)函數(shù)調(diào)用" << endl;
    }

class Son: public Base{
public:
    Son(int val):m_val(new int (val)) {
        cout << "Son 的構(gòu)造函數(shù)調(diào)用" << endl;
    }

    ~Son(){
        cout << "Son 的析構(gòu)函數(shù)調(diào)用" << endl;
        if (m_val != NULL) {
            
            delete m_val;
            cout << "Son 析構(gòu)函數(shù)的堆內(nèi)存釋放" << endl;
            m_val = NULL;
        }
    }
    void func(){
        cout << "Son func" << endl;
    }

    void funcTest(){
        cout << "funcTest 函數(shù)調(diào)用" << endl;
    }

    int* m_val = NULL;
};


void test() {
   
    Base *base = new Son(10);
    base->func();
    //base->funcTest(); //無(wú)法調(diào)用,因?yàn)樘摵瘮?shù)表中不能找到這個(gè)函數(shù)的地址
    delete base;
    base = NULL;
}


int main () {
    test();    
    system("pause");
    return 0;

}

代碼運(yùn)行結(jié)果為:

 可以看到只要把Base類(lèi)的析構(gòu)函數(shù)寫(xiě)成虛析構(gòu)函數(shù)或純虛析構(gòu)函數(shù),通過(guò)父類(lèi)指針調(diào)用函數(shù),子類(lèi)的析構(gòu)代碼會(huì)被調(diào)用,子類(lèi)堆區(qū)內(nèi)存得到釋放。

到此這篇關(guān)于C++虛函數(shù)表和虛析構(gòu)介紹的文章就介紹到這了,更多相關(guān)C++虛函數(shù)表和虛析構(gòu)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • OPENCV批量讀取圖片實(shí)現(xiàn)方法

    OPENCV批量讀取圖片實(shí)現(xiàn)方法

    下面小編就為大家?guī)?lái)一篇OPENCV批量讀取圖片實(shí)現(xiàn)方法。小編覺(jué)得挺不錯(cuò)的?,F(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-06-06
  • C語(yǔ)言中傳值與傳指針的介紹與區(qū)別

    C語(yǔ)言中傳值與傳指針的介紹與區(qū)別

    這篇文章主要給大家介紹了關(guān)于C語(yǔ)言中傳值與傳指針的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用C語(yǔ)言具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-06-06
  • c++?error:crosses?initialization?of問(wèn)題解決分析

    c++?error:crosses?initialization?of問(wèn)題解決分析

    這篇文章主要介紹了c++?error:crosses?initialization?ofde?問(wèn)題解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-08-08
  • OpenGL繪制三次Bezier曲線

    OpenGL繪制三次Bezier曲線

    這篇文章主要為大家詳細(xì)介紹了OpenGL繪制三次Bezier曲線,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-04-04
  • Qt qml實(shí)現(xiàn)動(dòng)態(tài)輪播圖效果

    Qt qml實(shí)現(xiàn)動(dòng)態(tài)輪播圖效果

    這篇文章主要為大家詳細(xì)介紹了Qt和qml實(shí)現(xiàn)動(dòng)態(tài)輪播圖效果的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,有需要的小伙伴可以參考一下
    2024-12-12
  • C++中Semaphore內(nèi)核對(duì)象用法實(shí)例

    C++中Semaphore內(nèi)核對(duì)象用法實(shí)例

    這篇文章主要介紹了C++中Semaphore內(nèi)核對(duì)象用法實(shí)例,有助于深入了解信號(hào)量(Semaphore)的基本用法,需要的朋友可以參考下
    2014-10-10
  • C++使用redis的實(shí)例詳解

    C++使用redis的實(shí)例詳解

    這篇文章主要介紹了C++使用redis的實(shí)例詳解的相關(guān)資料,希望通過(guò)本文能幫助到大家,讓大家理解掌握這部分內(nèi)容,需要的朋友可以參考下
    2017-10-10
  • CLion搭建配置C++開(kāi)發(fā)環(huán)境的圖文教程 (MinGW-W64 GCC-8.1.0)

    CLion搭建配置C++開(kāi)發(fā)環(huán)境的圖文教程 (MinGW-W64 GCC-8.1.0)

    這篇文章主要介紹了CLion搭建配置C++開(kāi)發(fā)環(huán)境的教程 (MinGW-W64 GCC-8.1.0),本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-02-02
  • C++ 虛函數(shù)專題

    C++ 虛函數(shù)專題

    這篇文章主要介紹了C++中虛函數(shù)的知識(shí)點(diǎn),文中配合代碼講解非常細(xì)致,供大家參考和學(xué)習(xí),感興趣的朋友可以了解下
    2020-06-06
  • 數(shù)據(jù)結(jié)構(gòu)之歸并排序的實(shí)例詳解

    數(shù)據(jù)結(jié)構(gòu)之歸并排序的實(shí)例詳解

    這篇文章主要介紹了數(shù)據(jù)結(jié)構(gòu)之歸并排序的實(shí)例詳解的相關(guān)資料,這里對(duì)歸并排序進(jìn)行詳細(xì)介紹,需要的朋友可以參考下
    2017-08-08

最新評(píng)論