今天在實(shí)現(xiàn)一個(gè)文件訪問的組件時(shí),發(fā)現(xiàn)始終有內(nèi)存泄漏;跟蹤后發(fā)現(xiàn),是兩個(gè) COM 對(duì)象互相引用導(dǎo)致計(jì)數(shù)器無法歸零導(dǎo)致的。
大致情況是: CDGFile 為主對(duì)象;CDGFileSegment 為子對(duì)象;他們之間互相保留指針,因此都作了引用計(jì)數(shù),現(xiàn)在,使用者通過智能指針各保留了一份引用,如下:
CDGFilePtr ptrFile;
CDGFileSegment ptrSegment;
此時(shí),ptrFile,ptrSegment 中的計(jì)數(shù)器均為 2 :因?yàn)橹悄苤羔樃鞅A袅艘环?;?duì)象之間也各保留了一份。當(dāng)退出當(dāng)前函數(shù)時(shí),智能指針先后析構(gòu),指針數(shù)先后減 1 ;但內(nèi)部互相引用的計(jì)數(shù)仍然存在,因此導(dǎo)致了內(nèi)存泄漏!
解決方法:當(dāng)需要互相引用時(shí),應(yīng)該根據(jù)邏輯上的層次,僅對(duì)一方作引用計(jì)數(shù);
比如本例的解決方案:
1、CDGFile 是 CDGFileSegment 的父對(duì)象,因此 CDGFileSegment 中保留的 CDGFile 不應(yīng)該作引用計(jì)數(shù),程序邏輯應(yīng)該自己控制父對(duì)象一直有效;
2、因?yàn)楦笇?duì)象 CDGFile 中保留有 CDGFileSegment 的指針,因此,CDGFile 在銷毀前,應(yīng)該將 CDGFileSegment 中保留的 CDGFile 指針置為空,以標(biāo)志父對(duì)象已失效!
3、在對(duì)象結(jié)束生命期之前,清除引用。例如,可以增加 Final() 成員函數(shù),在該函數(shù)中清除對(duì)其他對(duì)象的引用。
以下是網(wǎng)上找到的相關(guān)資料,做個(gè)記號(hào):
http://topic.csdn.net/u/20090705/04/de76dce2-031d-4566-b1b5-84380558328e.html