继承和多态二:虚析构函数
雖然我們已經(jīng)知道了什么是繼承和多態(tài),也明白了多態(tài)依賴于繼承,但是在多態(tài)中存在哪些問(wèn)題呢?
多態(tài)中可能存在的內(nèi)存泄露問(wèn)題
例如下面的程序中,在圓形Circle的類中定義一個(gè)圓心的坐標(biāo),并且坐標(biāo)是在堆中申請(qǐng)的內(nèi)存,則在mian函數(shù)中通過(guò)基類指針操作派生類對(duì)象的成員函數(shù)是沒有問(wèn)題的,可是在銷毀對(duì)象內(nèi)存的時(shí)候則只是執(zhí)行了基類的析構(gòu)函數(shù),派生類的析構(gòu)函數(shù)卻沒有執(zhí)行,這會(huì)導(dǎo)致內(nèi)存泄漏。換句話說(shuō)如果delete后邊跟基類的指針則只會(huì)執(zhí)行基類的析構(gòu)函數(shù),如果delete后面跟的是派生類的指針,那么它即會(huì)執(zhí)行派生類的析構(gòu)函數(shù),也會(huì)執(zhí)行基類的析構(gòu)函數(shù)
class Shape { public:virtual double calcArea(){...}//虛函數(shù) Shape();~Shape();.... private:.... }; Shape::Shape() { cout<<"class Shape was created"<<endl; } Shape::~Shape() { cout<<"class Shape was deleted"<<endl; } class Circle:public Shape { public:Circle(int x,int y,double r);~Circle();virtual double calcArea();//此處可不加virtual,但系統(tǒng)默認(rèn)加上 .... private:double m_dR;Coordinate *m_pCenter; //坐標(biāo)類指針 .... }; Circle::Circle(int x,int y,double r) {m_pCenter=new Coordinate(x,y);m_dR=r;cout<<"class Circle was created"<<endl; } Circle::~Circle() {delete m_pCenter;m_pCenter=NULL;cout<<"class Circle was deleted"<<endl; } .... int main() {Shape *shape1=new Circle(5,6,4.0);//基類對(duì)象,改成Circle *circle=new Circle(5,6,4.0)則為派生類對(duì)象shape1->calcArea();delete shape1;shape1=NULL;//派生類對(duì)象的內(nèi)存未回收return 0; } 打印結(jié)果可見,派生類的析構(gòu)函數(shù)并沒有被執(zhí)行,而只是回收了開辟給基類對(duì)象的空間,那如何解決這個(gè)問(wèn)題呢,固然可以通過(guò)delete派生類對(duì)象實(shí)現(xiàn),除此之外,還可以引入虛析構(gòu)函數(shù),你看普通的虛函數(shù)都可以實(shí)現(xiàn)多態(tài)的動(dòng)態(tài)聯(lián)編,那對(duì)于析構(gòu)函數(shù)呢?虛析構(gòu)函數(shù)也會(huì)在最后對(duì)象銷毀內(nèi)存時(shí)動(dòng)態(tài)鏈接到派生類對(duì)象。 可以這樣解釋它:如果基類當(dāng)中定義了虛析構(gòu)函數(shù),那么基類的虛函數(shù)表當(dāng)中就會(huì)有一個(gè)基類的虛析構(gòu)函數(shù)的入口指針,指向的是基類的虛析構(gòu)函數(shù),派生類的虛函數(shù)表當(dāng)中也會(huì)產(chǎn)生一個(gè)派生類的虛析構(gòu)函數(shù)的入口指針,指向的是派生類的虛析構(gòu)函數(shù),這個(gè)時(shí)候使用基類的指針指向派生類的對(duì)象,delete掉基類指針,就會(huì)通過(guò)指向的基類的對(duì)象找到基類的虛函數(shù)表指針,從而找到虛函數(shù)表,在虛函數(shù)表中找到派生類的虛析構(gòu)函數(shù),從而使得派生類的析構(gòu)函數(shù)得以執(zhí)行,派生類的析構(gòu)函數(shù)執(zhí)行之后系統(tǒng)會(huì)自動(dòng)執(zhí)行父類的虛析構(gòu)函數(shù)。即整個(gè)執(zhí)行過(guò)程是:virtual關(guān)鍵字可以修飾普通的成員函數(shù),也可以修飾析構(gòu)函數(shù),但并不是沒有限制
virtual在函數(shù)中的使用限制
-
- 普通函數(shù)不能是虛函數(shù),也就是說(shuō)這個(gè)函數(shù)必須是某一個(gè)類的成員函數(shù),不可以是一個(gè)全局函數(shù),否則會(huì)導(dǎo)致編譯錯(cuò)誤。
- 靜態(tài)成員函數(shù)不能是虛函數(shù) static成員函數(shù)是和類同生共處的,他不屬于任何對(duì)象,使用virtual也將導(dǎo)致錯(cuò)誤。
- 內(nèi)聯(lián)函數(shù)不能是虛函數(shù) 如果修飾內(nèi)聯(lián)函數(shù) 如果內(nèi)聯(lián)函數(shù)被virtual修飾,計(jì)算機(jī)會(huì)忽略inline使它變成存粹的虛函數(shù)。(inline將會(huì)在后面的博客中介紹)
- 構(gòu)造函數(shù)不能是虛函數(shù),否則會(huì)出現(xiàn)編譯錯(cuò)誤。
轉(zhuǎn)載于:https://www.cnblogs.com/cvtoEyes/p/8492247.html
總結(jié)
以上是生活随笔為你收集整理的继承和多态二:虚析构函数的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: Tomcat启动时项目重复加载,导致资源
- 下一篇: Vue-Router + Vuex 实现