當(dāng)前位置:首頁 > 芯聞號 > 充電吧
[導(dǎo)讀]本文目的是實現(xiàn)一個實用的對C++類計數(shù)的類,同時在實現(xiàn)過程中指出一些容易為人忽視的C++知識。要實現(xiàn)一個類的對象(實例)計數(shù),即程序運行中此類有多少個對象存在,最容易的實現(xiàn)方法是使用靜態(tài)數(shù)據(jù)成員。如下

本文目的是實現(xiàn)一個實用的對C++類計數(shù)的類,同時在實現(xiàn)過程中指出一些容易為人忽視的C++知識。

要實現(xiàn)一個類的對象(實例)計數(shù),即程序運行中此類有多少個對象存在,最容易的實現(xiàn)方法是使用靜態(tài)數(shù)據(jù)成員。如下:

?01.classWidget {02.public:03.Widget() { ++count; }04.Widget(constWidget&) { ++count; } 05.~Widget() { --count; }06.07.staticsize_thowMany() 08.{returncount; } 09.10.private:11.staticsize_tcount; 12.};13.14.//cpp文件中15.size_tWidget::count = 0;

注意構(gòu)造函數(shù)也要增加計數(shù),這一點很多人容易忘記。

但是如果程序中有多個需要實例計數(shù)的類,則在每個類中加入上面代碼未免繁瑣、易錯。這種情況下,最好是實現(xiàn)一個通用計數(shù)類。它應(yīng)該具備一下特點:

易于使用:任何需要計數(shù)的類(以下簡稱客戶類)只要添加少數(shù)代碼即可使用;

有效率:不增加客戶類大小,對客戶類性能沒有影響;

健壯:客戶類使用時,不容易誤用。

下面我們將逐步實現(xiàn)并完善這個通用的計數(shù)類。

?01.classCounter {02.public:03.Counter() { ++count; }04.Counter(constCounter&) { ++count; } 05.~Counter() { --count; }06.staticsize_thowMany() 07.{returncount; } 08.09.private:10.staticsize_tcount; 11.};12.13.// This still goes in an implementation file14.size_tCounter::count = 0;

上面這個Counter類能否正確完成計數(shù)呢?例如:Widget類利用它來進行實例計數(shù):

?01.// embed a Counter to count objects02.classWidget {03.public:04......// all the usual public05.// Widget stuff06.staticsize_thowMany() 07.{returnCounter::howMany(); }08.private:09......// all the usual private10.// Widget stuff11.Counter c;12.};13.14.//or:15.16.// inherit from Counter to count objects17.classWidget:public Counter { 18......// all the usual public19.// Widget stuff20.private:21......// all the usual private22.// Widget stuff23.};

對于Widget本身來說,Counter完成了任務(wù)。然而,如果我們在同一進程中還需要利用Counter來計數(shù)Fish類,顯然,Counter就不能勝任,因為它只有一個靜態(tài)成員變量,它會將Widget和Fish的個數(shù)一起統(tǒng)計。這個方案不行,怎么辦?用模板!如下:

?01.template02.classCounter {03.public:04.Counter() { ++count; }05.Counter(constCounter&) { ++count; } 06.~Counter() { --count; }07.08.staticsize_thowMany() 09.{returncount; } 10.11.private:12.staticsize_tcount; 13.};14.15.// this now can go in header16.templatesize_tCounter::count = 0;

則上面的實現(xiàn)變成:

?01.// embed a Counter to count objects02.classWidget {03.public:04......05.staticsize_thowMany() 06.{returnCounter::howMany();}07.private:08......09.Counter c;10.};11.12.//or:13.14.// inherit from Counter to count objects15.classWidget:public Counter { 16......17.};

這樣,其他類就可以使用Counter計數(shù)自己的實例了,它們將互不影響。

上面兩種方案都可正確實現(xiàn)計數(shù),我們繼續(xù)探討這兩種方案的優(yōu)缺點。

首先講public繼承,即class Widget: public Counter這種方案:有經(jīng)驗的讀者肯定會想到基類Counter的析構(gòu)函數(shù)要變?yōu)樘摵瘮?shù)。否則通過基類指針delete派生類時,結(jié)果未定義(可能導(dǎo)致程序crash或其他)

?1.Counter *pw =newWidget; // get base class ptr to derived class object2.......3.deletepw;// yields undefined results if the base class lacks a virtual destructor

但一旦Counter有虛析構(gòu)函數(shù),就會給類帶入vTable,多占用了空間并影響客戶類的效率。解決方法可以是將析構(gòu)函數(shù)作為protected成員。這樣就不能delete pw,因為它會導(dǎo)致編譯錯誤。

?1.template2.classCounter {3.public:4......5.protected:6.~Counter() { --count; }7......8.};

其次,Counter作為客戶類的成員變量這種方案(這時Counter的析構(gòu)函數(shù)必須public)。一個明顯的缺點是客戶類必須定義Counter為其成員變量同時還得定義一個inline函數(shù)以調(diào)用Counter類得HowMany函數(shù)。另一個較隱蔽的缺點:它增大了客戶類所占用的內(nèi)存。Counter類沒有非靜態(tài)成員變量,有人就可能會認為Counter對象的大小為0,其實不然,C++規(guī)定所有對象的大小最小必須為1字節(jié)。所以這用方案增加了客戶類的大小。使用派生則不一樣,基類size可以0,所以public繼承方案不會增加客戶類的大小。

除了上面兩種方案,還可以使用private繼承,即class Widget: private Counter。類似于第一種方案:

?1.classWidget:private Counter { 2.public:3.// make howMany public4.usingCounter::howMany;5.6......// rest of Widget is unchanged7.};

它直接防止下面的代碼:

?1.Counter *pw =newWidget; //私有繼承不允許這樣轉(zhuǎn)換

綜合看來,public繼承方案已經(jīng)比較完善了。然而,還是有些值得注意的地方。假如有另一個類SpecialWidget,其繼承于Widget,對類SpecialWidget的對象計數(shù)就只能如下:

?1.classSpecialWidget:publicWidget, 2.publicCounter {3.public:4.};

這樣,對SpecialWidget的對象計數(shù)是正確的,但對Widget對象的計數(shù)是錯誤的。這時Widget的計數(shù)是Widget類的所有對象SpecialWidget類的所有對象的總和。為什么?因為每創(chuàng)建一個SpecialWidget對象,Widget構(gòu)造函數(shù)就要調(diào)用一次,就增加一次計數(shù)。

總結(jié)

用模板實現(xiàn)的這個對象計數(shù)類可以滿足絕大多數(shù)需求,但不適用于計數(shù)有繼承關(guān)系的類。本文的核心思想來源于CUG上C++大師Scott Meyers的一篇文章并有所改動。

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

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

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

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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