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