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