欧美大屁股bbbbxxxx,狼人大香伊蕉国产www亚洲,男ji大巴进入女人的视频小说,男人把ji大巴放进女人免费视频,免费情侣作爱视频

歡迎來到入門教程網(wǎng)!

C語言

當(dāng)前位置:主頁 > 軟件編程 > C語言 >

詳解C++中虛析構(gòu)函數(shù)的作用及其原理分析

來源:本站原創(chuàng)|時間:2020-01-10|欄目:C語言|點(diǎn)擊: 次

C++中的虛析構(gòu)函數(shù)到底什么時候有用的,什么作用呢。

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

總的來說虛析構(gòu)函數(shù)是為了避免內(nèi)存泄露,而且是當(dāng)子類中會有指針成員變量時才會使用得到的。也就說虛析構(gòu)函數(shù)使得在刪除指向子類對象的基類指針時可以調(diào)用子類的析構(gòu)函數(shù)達(dá)到釋放子類中堆內(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;
}

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

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

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

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

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

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

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

(2)如果父類的析構(gòu)函數(shù)加virtual關(guān)鍵字
當(dāng)父類的析構(gòu)函數(shù)聲明成虛析構(gòu)函數(shù)的時候,當(dāng)子類繼承父類,父類的指針指向子類時,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),但是當(dāng)父類的構(gòu)造函數(shù)/析構(gòu)函數(shù)即使被聲明virtual,子類的構(gòu)造/析構(gòu)方法仍無法覆蓋父類的構(gòu)造方法和析構(gòu)方法。這是由于父類的構(gòu)造函數(shù)和析構(gòu)函數(shù)是子類無法繼承的,也就是說每一個類都有自己獨(dú)有的構(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)

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

以上所述是小編給大家介紹的C++中虛析構(gòu)函數(shù)的作用及其原理分析詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對我們網(wǎng)站的支持!

上一篇:C++實現(xiàn)新年賀卡程序

欄    目:C語言

下一篇:C++基于消息隊列的多線程實現(xiàn)示例代碼

本文標(biāo)題:詳解C++中虛析構(gòu)函數(shù)的作用及其原理分析

本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/368.html

網(wǎng)頁制作CMS教程網(wǎng)絡(luò)編程軟件編程腳本語言數(shù)據(jù)庫服務(wù)器

如果侵犯了您的權(quán)利,請與我們聯(lián)系,我們將在24小時內(nèi)進(jìn)行處理、任何非本站因素導(dǎo)致的法律后果,本站均不負(fù)任何責(zé)任。

聯(lián)系QQ:835971066 | 郵箱:835971066#qq.com(#換成@)

Copyright © 2002-2020 腳本教程網(wǎng) 版權(quán)所有