當(dāng)前位置:首頁(yè) > 芯聞號(hào) > 充電吧
[導(dǎo)讀]本條款開始前我要先闡述重點(diǎn)2:你不該在構(gòu)造函數(shù)和析構(gòu)函數(shù)期間調(diào)用virtual函數(shù),因?yàn)檫@樣的調(diào)用不會(huì)帶來(lái)你預(yù)想的結(jié)果,就算有你也不會(huì)高興。如果你同時(shí)也是一位Java 或C# 程序員,請(qǐng)更加注意本條款

本條款開始前我要先闡述重點(diǎn)2:你不該在構(gòu)造函數(shù)和析構(gòu)函數(shù)期間調(diào)用virtual函數(shù),因?yàn)檫@樣的調(diào)用不會(huì)帶來(lái)你預(yù)想的結(jié)果,就算有你也不會(huì)高興。如果你同時(shí)也是一位Java 或C# 程序員,請(qǐng)更加注意本條款,因?yàn)檫@是C++ 與它們不相同的一個(gè)地方。

? ? ? ?假設(shè)你有個(gè)class 繼承體系,用來(lái)塑模股市交易如買進(jìn)、賣出的訂單等等。這樣的交易一定要經(jīng)過(guò)審計(jì),所以每當(dāng)創(chuàng)建個(gè)個(gè)交易對(duì)象,在審計(jì)日志(audit log) 中也需要?jiǎng)?chuàng)建一筆適當(dāng)記錄。下面是一個(gè)看起來(lái)頗為合理的做法:

class Transaction {// 所有交易的baseclass  
public:  
    Transaction( );  
    virtual void logTransaction() const = 0;// 做出一份因類型不同而不同的日志記錄  
    ......  
};  
Transaction::Transaction ()                // base class 構(gòu)造函數(shù)之實(shí)現(xiàn)  
{  
    ......  
    logTransaction(); // 最后動(dòng)作是志記這筆交易  
}  
class BuyTransaction: public Transaction {  // derivedclass  
public:  
    virtual void logTransaction() const;    // 志記(log) 此型交易  
   ....  
} ;  
class SellTransaction: public Transaction { // derivedclass  
public:  
    virtual void logTransaction() const;    // 志記(log) 此型交易  
    ....  
};  

? ? ? ?現(xiàn)在,當(dāng)以下這行被執(zhí)行,會(huì)發(fā)生什么事:

BuyTransaction b;  
? ? ? ?無(wú)疑地會(huì)有一個(gè)BuyTransaction 構(gòu)造函數(shù)被調(diào)用,但首先Transaction 構(gòu)造函數(shù)一定會(huì)更早被調(diào)用;是的, derived class 對(duì)象內(nèi)的base class 成分會(huì)在derived class自身成分被構(gòu)造之前先構(gòu)造妥當(dāng)。Transaction 構(gòu)造函數(shù)的最后一行調(diào)用virtual 函數(shù)logTransaction,這正是引發(fā)驚奇的起點(diǎn)。這時(shí)候被調(diào)用的logTransaction 是Transaction 內(nèi)的版本,不是BuyTransaction 內(nèi)的版本一一即使目前即將建立的對(duì)象類型是BuyTransaction。是的, base class 構(gòu)造期間virtual 函數(shù)絕不會(huì)下降到derived classes 階層。取而代之的是,對(duì)象的作為就像隸屬base 類型一樣。非正式的說(shuō)法或許比較傳神:在base class 構(gòu)造期間, virtual 函數(shù)不是virtual 函數(shù)。
? ? ? ?這一似乎反直覺(jué)的行為有個(gè)好理由。由于base class 構(gòu)造函數(shù)的執(zhí)行更早于derived class 構(gòu)造函數(shù),當(dāng)base class 構(gòu)造函數(shù)執(zhí)行時(shí)derived class 的成員變量尚未初始化。如果此期間調(diào)用的virtual 函數(shù)下降至derived classes 階層,要知道derived class的函數(shù)幾乎必然取用local 成員變量,而那些成員變量尚未初始化。這將是一張通往不明確行為和徹夜調(diào)試大會(huì)串的直達(dá)車票。"要求使用對(duì)象內(nèi)部尚未初始化的成分"是危險(xiǎn)的代名詞,所以C++ 不讓你走這條路。

? ? ? ?其實(shí)還有比上述理由更根本的原因:在derived class 對(duì)象的base class 構(gòu)造期間,對(duì)象的類型是base class 而不是derived class。

? ? ? ? 相同道理也適用于析構(gòu)函數(shù)。一旦derived class 析構(gòu)函數(shù)開始執(zhí)行,對(duì)象內(nèi)的derived class 成員變量便呈現(xiàn)未定義值,所以C++ 視它們仿佛不再存在。進(jìn)入baseclass 析構(gòu)函數(shù)后對(duì)象就成為一個(gè)base class 對(duì)象,而C++ 的任何部分包括virtual 函數(shù)、dynamic casts 等等也就那么看待它。

? ? ? ?其他方案可以解決這個(gè)問(wèn)題。一種做法是在class Transaction 內(nèi)將logTransaction 函數(shù)改為non-virtual,然后要求derived class 構(gòu)造函數(shù)傳遞必要信息給Transaction 構(gòu)造函數(shù),而后那個(gè)構(gòu)造函數(shù)便可安全地調(diào)用non-virtual logTransaction。像這樣:

class Transaction {  
public:  
    explicit Transaction(const std::string& logInfo) ;  
    void logTransaction(const std::string& logInfo) const; // 如今是個(gè)non-virtual函數(shù)  
    ......  
};  
Transacton::Transaction(conststd::string& logInfo)  
{  
     logTransaction(logInfo);                             //如今是個(gè)non-virtual 調(diào)用  
    ......  
}  
class BuyTransaction: public Transaction {  
public:  
    BuyTransaction( parameters)  
    : Transaction(createLogString( parameters ))        //將log 信息傳給base class 構(gòu)造函數(shù)  
    { ......}   
    ......  
private:  
    static std::string createLogString( parameters );  
};  
? ? ? ? 換句話說(shuō)由于你無(wú)法使用virtual 函數(shù)從base classes 向下調(diào)用,在構(gòu)造期間,你可以藉由"令derived classes 將必要的構(gòu)造信息向上傳遞至base class 構(gòu)造函數(shù)"替換之而加以彌補(bǔ)。

? ? ?請(qǐng)注意本例之BuyTransaction 內(nèi)的private static 函數(shù)createLogString 的運(yùn)用。是的,比起在成員初值列(member initialization list) 內(nèi)給予base class 所需數(shù)據(jù),利用輔助函數(shù)創(chuàng)建一個(gè)值傳給base class 構(gòu)造函數(shù)往往比較方便(也比較可讀)。令此函數(shù)為static ,也就不可能意外指向"初期未成熟之BuyTransaction 對(duì)象內(nèi)尚未初始化的成員變量"。這很重要,正是因?yàn)?那些成員變量處于未定義狀態(tài)",所以"在base class 構(gòu)造和析構(gòu)期間調(diào)用的virtual 函數(shù)不可下降至derived classes" 。

需要記住的

在構(gòu)造和析構(gòu)期間不要調(diào)用virtual 函數(shù),因?yàn)檫@類調(diào)用從不下降至derived class(比起當(dāng)前執(zhí)行構(gòu)造函數(shù)和析構(gòu)函數(shù)的那層)。

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來(lái)越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來(lái)越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開幕式在貴陽(yáng)舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語(yǔ)權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉