C++成員函數(shù)后面加override問題
C++成員函數(shù)后面加override
class A { ? ? virtual void foo(); } class B :public A { ? ? ? ? void foo() override;? }
如果一個成員函數(shù)后面跟了一個override關(guān)鍵字,說明這個函數(shù)將重寫這個函數(shù),下面的方式也可以進(jìn)行重寫。
void foo();? virtual foo();?
但是后面加override算是一種聲明,此函數(shù)要重寫同名函數(shù),所以如果將函數(shù)的名字寫錯了比如寫成f00則會報錯,而下面的兩個卻不會,這樣也可以提醒代碼閱讀者這是一個重寫的函數(shù)。
override虛析構(gòu)函數(shù)使用技巧
#include <cstdio> class Base? { public: ?? ?~Base() { ::printf("base\n"); } }; class Derived? ? ? : public Base? { public: ?? ?~Derived() { ::printf("derived\n"); } };
Base *ptr = new Derived; delete ptr;
由于父類析構(gòu)函數(shù)不是虛函數(shù),因此編譯器只能找到父類析構(gòu)函數(shù),而無法通過續(xù)表找到子類析構(gòu)函數(shù),因此上例會造成內(nèi)存泄漏,子類中的數(shù)據(jù)成員沒法得到釋放。最重要的是這種行為并不會被編譯器察覺
因此effective c++建議,一個類一旦確認(rèn)要被繼承就應(yīng)該在其析構(gòu)函數(shù)前加上關(guān)鍵字virtual
因此得到下面的寫法
#include <cstdio> class Base { public: ?? ?virtual ~Base() { ::printf("base\n"); } }; class Derived? ? ? : public Base { public: ?? ?~Derived() { ::printf("derived\n"); } };
如此,就不會出現(xiàn)多態(tài)下的內(nèi)存泄漏問題。
但是!如果父類設(shè)計者忘記加關(guān)鍵字virtual,或者由于業(yè)務(wù)的需要,當(dāng)前類必須被子類實現(xiàn)某些功能( interface-implement ),豈不是就會再次造成泄漏?
因此,作為子類的設(shè)計者,有義務(wù)提醒父類設(shè)計者,缺少關(guān)鍵字virtual,從而避免內(nèi)存泄漏
C++11 新特性中的關(guān)鍵字override,編譯器會檢查基類中的虛函數(shù)和派生類中帶有override的虛函數(shù)有沒有相同的函數(shù)簽名,一旦不匹配便會報錯
因此子類設(shè)計者可以在其析構(gòu)函數(shù)后增加關(guān)鍵字override,一旦父類缺少關(guān)鍵字virtual就會被編譯器發(fā)現(xiàn)并報錯
#include <cstdio> class Base { public: ?? ?virtual ~Base() { ::printf("base\n"); } }; class Derived? ? ? : public Base { public: ?? ?~Derived() override { ::printf("derived\n"); } };
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
C語言實現(xiàn)數(shù)據(jù)結(jié)構(gòu)串(堆分配存儲表示法)實例詳解
這篇文章主要介紹了C語言實現(xiàn)數(shù)據(jù)結(jié)構(gòu)串(堆分配存儲表示法)實例詳解的相關(guān)資料,需要的朋友可以參考下2017-07-07C++?如何將Lambda轉(zhuǎn)換成函數(shù)指針
這篇文章主要介紹了C++?如何將Lambda轉(zhuǎn)換成函數(shù)指針,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-11-11