条款9:不要在构造和析构过程中调用virtual函数
如下是一個(gè)股票交易的例子:
1 class Transaction // 交易的基類(lèi) 2 { 3 public: 4 Transaction(); 5 virtual void logTransaction() const = 0; // 用于記錄交易日志 6 }; 7 Transaction::Transaction() 8 { 9 logTransaction(); // 調(diào)用虛函數(shù) 10 } 11 12 class BuyTransaction : public Transaction // 買(mǎi)進(jìn)股票 13 { 14 virtual void logTransaction() const; 15 }; 16 void BuyTransaction::logTransaction() const{ } 17 18 class SellTransaction : public Transaction // 賣(mài)出股票 19 { 20 virtual void logTransaction() const; 21 }; 22 void SellTransaction::logTransaction() const{ } 23 24 int main() 25 { 26 BuyTransaction b; 27 28 return 0; 29 }上述代碼中執(zhí)行BuyTransaction b時(shí),會(huì)調(diào)用基類(lèi)的構(gòu)造函數(shù),而基類(lèi)的構(gòu)造函數(shù)會(huì)調(diào)用一個(gè)虛函數(shù)來(lái)完成工作。但是該虛函數(shù)是否會(huì)調(diào)用派生類(lèi)的對(duì)應(yīng)函數(shù)呢?答案是否定的,解釋如下:
1>? 基類(lèi)的構(gòu)造是先于派生類(lèi)的構(gòu)造的。當(dāng)基類(lèi)的構(gòu)造函數(shù)執(zhí)行時(shí),派生類(lèi)中的成員尚未被初始化,如果允許基類(lèi)構(gòu)造期間調(diào)用派生類(lèi)的函數(shù),可能也會(huì)調(diào)用了沒(méi)有初始化的“垃圾值”,導(dǎo)致不確定行為發(fā)生。
2>? 基類(lèi)構(gòu)造期間,virtual函數(shù)的行為絕對(duì)不會(huì)伸到派生類(lèi)中,因?yàn)榇藭r(shí)構(gòu)造的是基類(lèi),即此時(shí)構(gòu)造的對(duì)象的類(lèi)型是基類(lèi)而非派生類(lèi),這點(diǎn)很重要。
?
但是有時(shí)我們?cè)跇?gòu)造基類(lèi)的時(shí)候又需要派生類(lèi)中才有的參數(shù)信息,如何解決:
在基類(lèi)中將logTransaction函數(shù)改為非虛函數(shù),然后在構(gòu)造派生類(lèi)時(shí)將必要的參數(shù)傳遞到基類(lèi)的構(gòu)造函數(shù)中去,像如下這樣:
1 #include <string> 2 3 class Transaction // 交易的基類(lèi) 4 { 5 public: 6 explicit Transaction(const std::string& logInfo); 7 void logTransaction(const std::string& logInfo) const; // 用于記錄交易日志 8 }; 9 Transaction::Transaction(const std::string& logInfo) 10 { 11 logTransaction(logInfo); // 調(diào)用非虛函數(shù)12 } 13 void Transaction::logTransaction(const std::string& logInfo) const{ } 14 15 class BuyTransaction : public Transaction // 買(mǎi)進(jìn)股票 16 { 17 public: 18 BuyTransaction(const std::string& parameter) : Transaction(parameter){ } 19 }; 20 21 22 int main() 23 { 24 BuyTransaction b("BT"); 25 26 return 0; 27 }
?
由上可知,如果基類(lèi)構(gòu)造時(shí)要使用派生類(lèi)的信息,可以通過(guò)派生類(lèi)構(gòu)造函數(shù)的初始化列表傳入相關(guān)參數(shù),切不可調(diào)妄圖調(diào)用虛函數(shù)實(shí)現(xiàn),因?yàn)樵跇?gòu)造基類(lèi)期間其對(duì)象的類(lèi)型是基類(lèi)類(lèi)型。
對(duì)于析構(gòu)函數(shù)也是同樣的道理,基類(lèi)的析構(gòu)總是在派生類(lèi)析構(gòu)后才進(jìn)行,此時(shí)派生類(lèi)的對(duì)象已經(jīng)不存在了,而調(diào)用虛函數(shù)必然會(huì)引發(fā)一個(gè)不確定行為。當(dāng)然你也可以將基類(lèi)的虛函數(shù)實(shí)現(xiàn)后在析構(gòu)函數(shù)中進(jìn)行調(diào)用,但是這樣的話,為什么不定義為非虛函數(shù)呢,虛函數(shù)的意義將不復(fù)存在了。
?
轉(zhuǎn)載于:https://www.cnblogs.com/benxintuzi/p/4527948.html
總結(jié)
以上是生活随笔為你收集整理的条款9:不要在构造和析构过程中调用virtual函数的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: Redis系列(四)-低成本高可用方案设
- 下一篇: 如何给easyui datagrid t