C++箴言:避開析構(gòu)函數(shù)調(diào)用虛函數(shù)
發(fā)表時間:2024-02-18 來源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要]如果你已經(jīng)從另外一種語言如C#或者Java轉(zhuǎn)向了C++,你會覺得,避免在類的構(gòu)造函數(shù)或者析構(gòu)函數(shù)中調(diào)用虛函數(shù)這一原則有點違背直覺。但是在C++中,違反這個原則會給你帶來難以預(yù)料的后果和無盡的煩惱。 正文 我想以重復(fù)本文的主題開篇:不要在類的構(gòu)造或者析構(gòu)函數(shù)中調(diào)用虛函數(shù),因為這種調(diào)用不會如你...
如果你已經(jīng)從另外一種語言如C#或者Java轉(zhuǎn)向了C++,你會覺得,避免在類的構(gòu)造函數(shù)或者析構(gòu)函數(shù)中調(diào)用虛函數(shù)這一原則有點違背直覺。但是在C++中,違反這個原則會給你帶來難以預(yù)料的后果和無盡的煩惱。
正文
我想以重復(fù)本文的主題開篇:不要在類的構(gòu)造或者析構(gòu)函數(shù)中調(diào)用虛函數(shù),因為這種調(diào)用不會如你所愿,即使成功一點,最后還會使你沮喪不已。如果你以前是一個Java或者C#程序員,請密切注意本節(jié)的內(nèi)容-這正是C++與其它語言的大區(qū)別之一。
假設(shè)你有一個為股票交易建模的類層次結(jié)構(gòu),例如買單,賣單,等等。為該類交易建立審計系統(tǒng)是非常重要的,這樣的話,每當創(chuàng)建一個交易對象,在審計登錄項上就生成一個適當?shù)娜肟陧。這看上去不失為一種解決該問題的合理方法:
class Transaction {// 所有交易的基類
public:
Transaction();
virtual void logTransaction() const = 0;//建立依賴于具體交易類型的登錄項
...
};
Transaction::Transaction() //實現(xiàn)基類的構(gòu)造函數(shù)
{
...
logTransaction(); //最后,登錄該交易
}
class BuyTransaction: public Transaction {
// 派生類
public:
virtual void logTransaction() const; //怎樣實現(xiàn)這種類型交易的登錄?
...
};
class SellTransaction: public Transaction {
//派生類
public:
virtual void logTransaction() const; //怎樣實現(xiàn)這種類型交易的登錄?
...
};
現(xiàn)在,請分析執(zhí)行下列代碼調(diào)用時所發(fā)生的事情:
BuyTransaction b;
很明顯,一個BuyTransaction類構(gòu)造器被調(diào)用。但是,首先調(diào)用的是Transaction類的構(gòu)造器-派生類對象的基類部分是在派生類部分之前被構(gòu)造的。Transaction構(gòu)造器的最后一行調(diào)用了虛函數(shù)logTransaction,但是奇怪的事情正是在此發(fā)生的。被調(diào)用函數(shù)logTransaction的版本是Transaction中的那個,而不是BuyTransaction中的那個-即使現(xiàn)在產(chǎn)生的對象的類型是BuyTransaction,情況也是如此。在基類的構(gòu)造過程中,虛函數(shù)調(diào)用從不會被傳遞到派生類中。代之的是,派生類對象表現(xiàn)出來的行為好象其本身就是基類型。不規(guī)范地說,在基類的構(gòu)造過程中,虛函數(shù)并沒有被"構(gòu)造"。
對上面這種看上去有點違背直覺的行為可以用一個理由來解釋-因為基類構(gòu)造器是在派生類之前執(zhí)行的,所以在基類構(gòu)造器運行的時候派生類的數(shù)據(jù)成員還沒有被初始化。如果在基類的構(gòu)造過程中對虛函數(shù)的調(diào)用傳遞到了派生類,派生類對象當然可以參照引用局部的數(shù)據(jù)成員,但是這些數(shù)據(jù)成員其時尚未被初始化。這將會導(dǎo)致無休止的未定義行為和徹夜的代碼調(diào)試。沿類層次往下調(diào)用尚未初始化的對象的某些部分本來就是危險的,所以C++干脆不讓你這樣做。