兩個程序員遇到的最普遍并且最難處理的問題是覆蓋分配內存的尾部以及內存泄露(在他們不需要內存的時候不能釋放內存)。
DEBUG
堆提供了強大的工具來解決這類內存分配的問題。
?
堆函數的調試版本
?
堆函數的調試版本調用
Release
創建中的標準的或基礎版本。當你請求一個內存塊的時候,調試堆管理器從基礎堆中分配一個稍微比你請求大一點的內存快,并且返回一個指針指向你的塊的部分。例如,如果你的程序包含調用:
malloc(10)
,在
Release
創建的時候,
malloc
將調用基礎堆分配程序來請求分配
10
個字節的空間;在
Debug
版本中,
malloc
將調用
malloc_dbg
,它將調用基礎堆來分配一個你請求的
10
個字節再加
36
個字節的額外內存。所有的在
debug
堆中的內存塊用一個單一的鏈表來連接,順序是按照分配的順序。
調試堆程序分配的額外內存用來保存信息,用于將調式內存塊連接在一起的指針和在你數據另一邊的小緩沖區來捕獲分配區域的越界。
當前,塊頭結構用來存儲調試堆的輔助信息定義為下面的形式(
DBGINT.H
):
typedef struct _CrtMemBlockHeader
{
// Pointer to the block allocated just before this one:
?? struct _CrtMemBlockHeader *pBlockHeaderNext;
// Pointer to the block allocated just after this one:
?? struct _CrtMemBlockHeader *pBlockHeaderPrev;
?? char *szFileName;??? // File name
?? int nLine;?????????? // Line number
?? size_t nDataSize;??? // Size of user block
?? int nBlockUse;?????? // Type of block
?? long lRequest;?????? // Allocation number
// Buffer just before (lower than) the user's memory:
?? unsigned char gap[nNoMansLandSize];
} _CrtMemBlockHeader;
?
/* In an actual memory block in the debug heap,
?* this structure is followed by:
?*?? unsigned char data[nDataSize];
?*?? unsigned char anotherGap[nNoMansLandSize];
?*/
?
NoMansLan
緩沖區在用戶數據區域的任意一邊是
4
個字節的大小,填充一個已知的值,用于調試堆程序來驗證用戶的內存沒有被越界的限制。調試堆也用已知的值來填充新的內存塊;如果你選擇將
FREED
塊保存在堆的連表中(如下面解釋),這些塊也會被填充已知的值。當前,實際字節值如下:
NoMansLand(0XFD)
這種緩沖區在內存的任意一邊,當前填充為
0XFD
,應用程序使用
Freed blocks(0XDD)
該塊在
_CRTDBG_DELAY_FREE_MEM_DF
標志被設置的時候始終保持在堆的連接連表中不使用;當前填充
0XDD
。
New object(OXCD)
當他們被分配的時候填充
0XCD