溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

C++中虛析構(gòu)函數(shù)的作用是什么

發(fā)布時間:2021-06-16 14:42:30 來源:億速云 閱讀:122 作者:Leah 欄目:編程語言

C++中虛析構(gòu)函數(shù)的作用是什么,針對這個問題,這篇文章詳細介紹了相對應(yīng)的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。

一.虛析構(gòu)函數(shù)的作用

總的來說虛析構(gòu)函數(shù)是為了避免內(nèi)存泄露,而且是當子類中會有指針成員變量時才會使用得到的。也就說虛析構(gòu)函數(shù)使得在刪除指向子類對象的基類指針時可以調(diào)用子類的析構(gòu)函數(shù)達到釋放子類中堆內(nèi)存的目的,而防止內(nèi)存泄露的.

我們知道,用C++開發(fā)的時候,用來做基類的類的析構(gòu)函數(shù)一般都是虛函數(shù)??墒牵瑸槭裁匆@樣做呢?下面用一個小例子來說明:

#include<iostream>
using namespace std;

class ClxBase
{
  public:
    ClxBase() {};
    virtual ~ClxBase() { cout<<"delete ClxBase"<<endl; };

    virtual void DoSomething() { cout << "Do something in class ClxBase!" << endl; };

};

class ClxDerived : public ClxBase
{
  public:
    ClxDerived() {};
    ~ClxDerived() { cout << "Output from the destructor of class ClxDerived!" << endl; };

    void DoSomething() { cout << "Do something in class ClxDerived!" << endl; };

};

int main(int argc, char const* argv[])
{
   ClxBase *pTest = new ClxDerived;
   pTest->DoSomething();
   delete pTest;
  return 0;
}

C++中虛析構(gòu)函數(shù)的作用是什么

但是,如果把類ClxBase析構(gòu)函數(shù)前的virtual去掉,那輸出結(jié)果就是下面的樣子了:

C++中虛析構(gòu)函數(shù)的作用是什么

沒有調(diào)動子類的析構(gòu)函數(shù)

也就是說,類ClxDerived的析構(gòu)函數(shù)根本沒有被調(diào)用!一般情況下類的析構(gòu)函數(shù)里面都是釋放內(nèi)存資源,而析構(gòu)函數(shù)不被調(diào)用的話就會造成內(nèi)存泄漏。我想所有的C++程序員都知道這樣的危險性。當然,如果在析構(gòu)函數(shù)中做了其他工作的話,那你的所有努力也都是白費力氣。

所以,文章開頭的那個問題的答案就是--這樣做是為了當用一個基類的指針刪除一個派生類的對象時,派生類的析構(gòu)函數(shù)會被調(diào)用。

當然,并不是要把所有類的析構(gòu)函數(shù)都寫成虛函數(shù)。因為當類里面有虛函數(shù)的時候,編譯器會給類添加一個虛函數(shù)表,里面來存放虛函數(shù)指針,這樣就會增加類的存儲空間。所以,只有當一個類被用來作為基類的時候,才把析構(gòu)函數(shù)寫成虛函數(shù)。

總結(jié)一下虛析構(gòu)函數(shù)的作用:

(1)如果父類的析構(gòu)函數(shù)不加virtual關(guān)鍵字
當父類的析構(gòu)函數(shù)不聲明成虛析構(gòu)函數(shù)的時候,當子類繼承父類,父類的指針指向子類時,delete掉父類的指針,只調(diào)動父類的析構(gòu)函數(shù),而不調(diào)動子類的析構(gòu)函數(shù)。

(2)如果父類的析構(gòu)函數(shù)加virtual關(guān)鍵字
當父類的析構(gòu)函數(shù)聲明成虛析構(gòu)函數(shù)的時候,當子類繼承父類,父類的指針指向子類時,delete掉父類的指針,先調(diào)動子類的析構(gòu)函數(shù),再調(diào)動父類的析構(gòu)函數(shù)。

二.虛析構(gòu)函數(shù)的原理分析

#include<iostream>
using namespace std;

class Base
{
public:
  Base(){cout<<"create Base"<<endl;}
  virtual ~Base(){cout<<"delete Base"<<endl;}
};

class Der : public Base
{
public:
  Der(){cout<<"create Der"<<endl;}
  ~Der(){cout<<"Delete Der"<<endl;}
};
int main(int argc, char const* argv[])
{
  Base *b = new Der;
  delete b;

  return 0;
}

從創(chuàng)建講起,用gdb調(diào)試你會發(fā)現(xiàn),

(1)先調(diào)用父類的構(gòu)造函數(shù),再調(diào)用子類的構(gòu)造函數(shù),

這里有一個問題:父類的構(gòu)造函數(shù)/析構(gòu)函數(shù)與子類的構(gòu)造函數(shù)/析構(gòu)函數(shù)會形成多態(tài),但是當父類的構(gòu)造函數(shù)/析構(gòu)函數(shù)即使被聲明virtual,子類的構(gòu)造/析構(gòu)方法仍無法覆蓋父類的構(gòu)造方法和析構(gòu)方法。這是由于父類的構(gòu)造函數(shù)和析構(gòu)函數(shù)是子類無法繼承的,也就是說每一個類都有自己獨有的構(gòu)造函數(shù)和析構(gòu)函數(shù)。

(2)而由于父類的析構(gòu)函數(shù)為虛函數(shù),所以子類會在所有屬性的前面形成虛表,而虛表內(nèi)部存儲的就是父類的虛函數(shù)

即使子類也有虛函數(shù),但是由于是單繼承,所以也只有一張?zhí)摫恚@在上一篇博客多態(tài)中講到過。
執(zhí)行 Base *b = new Der;之后b的最終形態(tài)

C++中虛析構(gòu)函數(shù)的作用是什么

(3)當delete父類的指針時,由于子類的析構(gòu)函數(shù)與父類的析構(gòu)函數(shù)構(gòu)成多態(tài),所以得先調(diào)動子類的析構(gòu)函數(shù);之所以再調(diào)動父類的析構(gòu)函數(shù),是因為delete的機制所引起的,delete 父類指針所指的空間,要調(diào)用父類的析構(gòu)函數(shù)。
所以結(jié)果就是這樣

C++中虛析構(gòu)函數(shù)的作用是什么

關(guān)于C++中虛析構(gòu)函數(shù)的作用是什么問題的解答就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注億速云行業(yè)資訊頻道了解更多相關(guān)知識。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

c++
AI