[原創(chuàng)文章歡迎轉(zhuǎn)載,但請(qǐng)保留作者信息]
Justin 于 2009-12-07
因?yàn)闇?zhǔn)備英文考試以及工作變動(dòng)的事情,已經(jīng)隔了很久沒(méi)有讀書(shū),也就很久沒(méi)有記錄筆記了。學(xué)習(xí)可以被打斷,但是不能終止。嗯不廢話了。繼續(xù)來(lái)讀圣賢書(shū)。
Item13 被安排在第三章“資源管理”的第一位,所以講得也很通俗易懂:要利用對(duì)象來(lái)管理資源(可以是分配出來(lái)的內(nèi)存、互斥量等)。這樣做的好處是可以避免因?yàn)閷?xiě)代碼時(shí)的疏忽而導(dǎo)致的資源流失(可以是內(nèi)存的泄漏,也可能是忘了解鎖)。為什么用對(duì)象來(lái)解決這個(gè)問(wèn)題呢?因?yàn)橐韵聝牲c(diǎn):
-
將釋放資源的代碼放在對(duì)象的析構(gòu)函數(shù)中,只要對(duì)象被析構(gòu),就可以保證資源被完整釋放。
-
C++的析構(gòu)函數(shù)自動(dòng)調(diào)用機(jī)制(C++’s automatic destructor invocation)能夠保證當(dāng)對(duì)象脫離控制時(shí)該對(duì)象的析構(gòu)函數(shù)被調(diào)用(比如一個(gè)對(duì)象在某函數(shù)內(nèi)部定義,那么在函數(shù)執(zhí)行結(jié)束后,這個(gè)對(duì)象的析構(gòu)函數(shù)會(huì)被自動(dòng)調(diào)用以銷(xiāo)毀此對(duì)象)。
是個(gè)好辦法,至少對(duì)我這樣時(shí)常丟三落四的人來(lái)說(shuō)是個(gè)福音。何況這樣的對(duì)象大多數(shù)情況下不用自己寫(xiě),auto_ptr(智能指針, smart pointer的一種)就可以勝任。
下面一段代碼其實(shí)和書(shū)上的沒(méi)有區(qū)別,只是為了加深印象
//
blah?blah
std::auto_ptr
<
void
*>
?pMem(AllocAndInitMemory());
//
feel?free?to?use?pMem
//
and?no?need?to?bother?releasing?the?memory
//
isn't?it?great?
高亮的部分即說(shuō)明了auto_ptr的用法,也附帶示范了所謂的RAII(Resource Acquisition Is Initialization)。我想這里可以翻譯為“資源申請(qǐng)時(shí)初始化”,雖然拗口了一些,但至少好理解,也不用再多做解釋。
這里的auto_ptr就像是個(gè)陪女友逛街的可憐家伙,女朋友(用戶)說(shuō)要什么,auto_ptr就乖乖的去買(mǎi)來(lái)(申請(qǐng)),想要用的時(shí)候就趕緊拿出來(lái)給女友用,哪怕她忘了曾經(jīng)買(mǎi)了某件衣服,auto_ptr還是會(huì)老老實(shí)實(shí)地送回家里,不用擔(dān)心會(huì)弄丟或是搞臟。嗯,有點(diǎn)做上帝的感覺(jué)吧?
不過(guò),沒(méi)有什么是十全十美的。大師馬上就說(shuō)到了用對(duì)象管理資源的問(wèn)題,好吧,應(yīng)該說(shuō)是需要注意的地方:
首先不能用一個(gè)以上的對(duì)象,比如說(shuō)auto_ptr,管理同一個(gè)資源。這個(gè)很好理解,就像一個(gè)漂亮的女孩子腳踏多條船(同時(shí)定義了多個(gè)auto_ptr),但是她必須要小心,不能讓多個(gè)對(duì)象參加同一個(gè)約會(huì):今天逛街的時(shí)候喊上了小張就不能叫小王,明天一起吃飯如果約了小李就別再和小趙定時(shí)間:一個(gè)資源一次只能讓一個(gè)對(duì)象來(lái)管理。
這樣的規(guī)則用在auto_ptr上就是書(shū)中奇怪的“=”行為:兩個(gè)auto_ptr對(duì)象A和B,A=B運(yùn)算的結(jié)果是A接管了B管理的資源,B指向的是null。(如果記不起來(lái)了,具體代碼見(jiàn)書(shū))這樣的行為很像是傳遞:小李陪女朋友逛完街后,送她到人民廣場(chǎng),在那里小陳約好了和她一起看電影。到了人民廣場(chǎng)之后小陳摟著女人開(kāi)心的走了,只剩下小李一個(gè)孤單的背影,杯具啊杯具……
為了避開(kāi)這種尷尬且極易被誤解的“=”操作,出現(xiàn)了以shared_ptr為代表的reference-counting smart pointer(RCSP)(計(jì)數(shù)智能指針?)。它們可以“共事一夫”,由一個(gè)公用的reference counter來(lái)計(jì)數(shù),某個(gè)shared_ptr在準(zhǔn)備拋棄一個(gè)資源時(shí)先參考reference counter的值,如果非零,說(shuō)明還有其他的“姐妹”在罩著資源,不需要也不可以釋放該資源;如果值為零,說(shuō)明當(dāng)前只有她一個(gè)人了,于是真正的擔(dān)當(dāng)起smart pointer的責(zé)任:釋放資源。
最后作者貌似意識(shí)到寫(xiě)得有點(diǎn)跑題,就提醒了一把:重點(diǎn)不在于是用auto_ptr還是shared_ptr還是其他的什么東東,而是要領(lǐng)會(huì)精神——使用對(duì)象來(lái)管理資源。比如說(shuō)用string對(duì)象來(lái)管理一個(gè)字符串而不是手工申請(qǐng)一片內(nèi)存然后用個(gè)指針“吧唧”粘上去就開(kāi)始用了。