檢測內存洩漏的主要工具是調試器和 CRT 調試堆函數。若要啟用調試堆函數,請在程序中包括以下語句:
#define CRTDBG_MAP_ALLOC
#include <stdlib.h>
#include <crtdbg.h>
注意 #include 語句必須采用上文所示順序。如果更改了順序,所使用的函數可能無法正確工作。
通過包括 crtdbg.h,將 malloc 和 free 函數映射到其“Debug”版本 _malloc_dbg 和 _free_dbg,這些函數將跟蹤內存分配和釋放。此映射只在調試版本(在其中定義了 _DEBUG)中發生。發布版本使用普通的 malloc 和 free 函數。
#define 語句將 CRT 堆函數的基版本映射到對應的“Debug”版本。並非絕對需要該語句,但如果沒有該語句,內存洩漏轉儲包含的有用信息將較少。
在添加了上面所示語句之後,可以通過在程序中包括以下語句來轉儲內存洩漏信息:
_CrtDumpMemoryLeaks();
當在調試器下運行程序時,_CrtDumpMemoryLeaks 將在“輸出”窗口中顯示內存洩漏信息。內存洩漏信息如下所示:
Detected memory leaks!Dumping objects ->C:PROGRAM FILESVISUAL STUDIOMyProjectsleaktestleaktest.cpp(20) : {18} normal block at 0x00780E80, 64 bytes long. Data: < > CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD CDObject dump complete.
如果不使用 #define _CRTDBG_MAP_ALLOC 語句,內存洩漏轉儲如下所示:
Detected memory leaks!Dumping objects ->{18} normal block at 0x00780E80, 64 bytes long. Data: < > CD CD CD CD CD CD CD CD CD CD CD CD CD CD CD CDObject dump complete.
未定義 _CRTDBG_MAP_ALLOC 時,所顯示的會是:
內存分配編號(在大括號內)。
塊類型(普通、客戶端或 CRT)。
十六進制形式的內存位置。
以字節為單位的塊大小。
前 16 字節的內容(亦為十六進制)。
定義了 _CRTDBG_MAP_ALLOC 時,還會顯示在其中分配洩漏的內存的文件。文件名後括號中的數字(本示例中為 20)是該文件內的行號。
轉到源文件中分配內存的行
在“輸出”窗口中雙擊包含文件名和行號的行。
- 或 -
在“輸出”窗口中選擇包含文件名和行號的行,然後按 F4 鍵。
_CrtSetDbgFlag
如果程序總在同一位置退出,則調用 _CrtDumpMemoryLeaks 足夠方便,但如果程序可以從多個位置退出該怎麼辦呢?不要在每個可能的出口放置一個對 _CrtDumpMemoryLeaks 的調用,可以在程序開始包括以下調用:
_CrtSetDbgFlag ( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );
該語句在程序退出時自動調用 _CrtDumpMemoryLeaks。必須同時設置 _CRTDBG_ALLOC_MEM_DF 和 _CRTDBG_LEAK_CHECK_DF 兩個位域,如上所示。
設置 CRT 報告模式
默認情況下,_CrtDumpMemoryLeaks 將內存洩漏信息轉儲到“輸出”窗口的“調試”窗格,如上所述。可以使用 _CrtSetReportMode 重置該設置,以轉儲到另一位置。如果使用庫,它可以將輸出重置到另一位置。在此情況下,可以使用以下語句將輸出位置設置回“輸出”窗口:
_CrtSetReportMode( _CRT_ERROR, _CRTDBG_MODE_DEBUG );
有關使用 _CrtSetReportMode 將輸出發送到其他位置的信息,請參見 _CrtSetReportMode。