VLD簡介
由於C/C++語言沒有所謂的垃圾收集器,內存的分配和釋放都需要程序員自己來控制,這會給C/C++程序員帶來一定的困難。當您的程序越來越復雜時,它的內存管理也會變得越來越困難。內存泄漏、內存越界是最常見的內存問題之一。
內存泄漏如果不是很嚴重的話,在短時間內對程序不會造成太大的影響,而且在進程終止的時候,所有分配的內存都會釋放掉。但是對於長時間運行的程序,其破壞力是驚人的,從性能下降到內存耗盡,甚至會影響到其它程序的正常運行。
此外,內存問題存在一個共同的特點,它本身並不會有很明顯的現象,當有異常出現時就很難檢查問題的原因所在,這給調試內存問題帶來了很大的難度。
VLD是一款用於VisualC++的免費內存泄漏檢查工具。可以在codeproject.com網站上找到,相比其它的內存泄漏哦給你根據,他在檢查內存泄漏的同事,還具有如下特點:
1) 可以得到內存泄漏點的調用堆棧,如果可以的話,還可以得到其所在的文件及行號;
2) 可以得到泄漏內存的完整數據;
3) 惡意設置內存泄漏報告的級別;
4) 它以動態庫的形式提供,無需編譯源代碼,只需要很小的改動程序;
5) 源代碼使用GNU許可發布,並有詳細的文檔及其注釋。
從使用的角度講,VLD簡單易用,對於使用者自己的代碼中唯一需要修改的地方是#include VLD的頭文件后正常運行自己的程序就可以發現內存問題。從研究角度上講,如果輸入到VLD源代碼,可以學習到堆內存分片與釋放的原理、內存檢查的原理機器內存操作的常用技巧等。
VLD使用
VLD網址:http://vld.codeplex.com/
http://www.codeproject.com/Articles/9815/Visual-Leak-Detector-Enhanced-Memory-Leak-Detectio
下載Visual LeakDetector,當前版本2.2.3,打開Visual C++ IDE的"工具"→"選項"→"項目和解決方案"→"VC++ 目錄",在"包含文件"中增加VLD的頭文件路徑"\include"路徑,在"庫文件"增加VLD庫文件的"\lib\Win32"路徑,此外動態庫的"\bin\Win32"路徑在安裝時已經添加到環境變量里面了,若是未添加,則需要手動拷貝"\bin\Win32"下的文件到可執行文件所在的目錄中(拷貝的文件有dbghelp.dll/Microsoft.DTfW.DHL.manifest/vld_x86.dll/vld.ini)。
接下來需要將VLD加入到自己的代碼中。方法很簡單,只要在包含入口函數的.cpp文件中包含vld.h就可以。如果這個cpp文件中包含了stdafx.h,則將包含vld.h的語句放在stdafx.h的包含語句之后,否則放在最前面。
示例程序:
#include<vld.h> // 包含VLD的頭文件
#include<stdlib.h>
#include<stdio.h>
void f()
{
int *p = new int(0x12345678);
printf("p=%08x, ", p);
}
int main()
{
f();
return 0;
}
注:VLD只能在Windows下使用,在包含vld.h頭文件時增加預編譯選項。
注:在Release模式下,不會鏈接VisualLeak Detector。
注:Visual LeakDetector有一些配置項,可以設置內存泄露報告的保存地(文件、調試器),拷貝"\Visual Leak Detector"路徑下的vld.ini文件到執行文件所在的目錄下(在IDE運行的話,則需要拷貝到工程目錄下),修改以下項:
ReportFile =.\memory_leak_report.txt
ReportTo = both
VLD工具原理
下面我們來看看VLD是如何工作的。在VisualC++中內置工具CRT Debug Heap工具,在使用Debug版本分配內存時,它會在內存塊中記錄分配該內存的文件名和行號。當程序退出時CRT會在main函數返回時做一些清理工作,此時檢查調試堆內存,如果仍然有內存沒釋放,則一定存在內存泄漏問題。從這些沒有被釋放的內存塊的頭中可以得到文件名和行號。這種靜態的方法可以檢查出內存泄漏,但是不知道泄漏究竟是怎么發生的,也不知道該內存分配語句是如何被執行到的,想要了解這些必須對內存分配過程進行動態跟蹤。VLD就是這樣做的,在每次內存分配的時候記錄其上下文,當程序退出時對檢測到的內存泄漏查找其上下文信息,並轉換成報告輸出到Output中。
初始化
VLD要記錄每次的內存分配,它通過Windows提供的分配鈎子allocation hooks來監視調試堆內存的分配。它是一個用戶自定義的回調函數,在每次從堆中分配內存之前被調用,在初始化是VLD使用_CrtSetAllocation注冊這個鈎子函數。
全局變量在程序初始化時就初始化,如果將VLD作為一個全局變量就可以與程序一起啟動,但是C/C++並沒有約定全局變量初始化的順序,如果其它全局變量的構造函數中有內存分配則可能無法檢測到。因此,VLD使用C/C++提供的#pragma init_seg來減少其它全局變量在它之前進行初始化。根據#pragma init_seg的定義,全局變量初始化分為3個階段,首先是compiler階段,一般進行C語言運行時庫的初始化;然后是lib段,一般用於第三方類庫的初始化扽;最后是user段,大部分的初始化都在這個階段進行。
記錄內存分配
一個內存分配鈎子函數需要具有如下的定義:
int AllocHook(int allocType, void*userData, size_t size,int blockType, long requestNumber, onst unsigned char*filename, int lineNumber);
該函數需要在VLD初始化時被注冊,每次從堆中分配內存前被調用,它需要處理的事情就是記錄下此時的調用堆棧和此時堆內存分配的唯一標識requestNumber。
得到當前堆棧的二進制表示並不是很復雜的事情,但是因為不同的體系結構、不同的編譯器、不同的操作系統所產生的堆棧內容是不一樣的,要解釋堆棧並得到整個函數的調用過程比較復雜。不過Windows提供了一個StackWalk64函數可以獲得堆棧的內容。
VLD是常用的C/C++內存泄漏檢查工具,可以在ViusalC++中使用,在Viusal Studio 2008和2010中使用需要注意兩點:
1) 版本問題:VLD已經更新到2.2版本,修正了許多bug,而且在2010版本下工作良好,VisualC++ 6.0推薦使用1.0版本,1.9b版本不是很穩定不建議使用,2.2版本的下載網址為http://vld.codeplex.com.
2) 設置變化:VC++Directories設置已經變化位置,在2010中設置過程如下:
View | Other Window | Property Manager
Go to "VC++ Directories" settings
Set include folder path
Set lib folder path
點OK,我們就設置好了include和lib目錄。
使用問題
問題1:VLD 1.9
在vista下使用vld的使用,總是出現錯誤無法正常工作,后來經過搜索,在http://www.codeproject.com/KB/applications/visualleakdetector.aspx
上的評論中找到了解決的方法:
評論“Solution forrunning 1.9 beta on Visual Studio 2008 with Vista ”給出了解決方法:
評論1:
VLD keptcrashing when trying to use 1.9g beta on Windows Vista, visual studio 2008. Itried all the suggestions on here and nothing worked. But I finally figured itout.
when you make a project in visual C++ 2008,it sets some strange advanced Linker properties that cause VLD to crash:
I changedLinker->Advanced->Randomized Base Address from Enable Image Randomization(/DYNAMICBASE) to Disable Image Randomization (/DYNAMICBASE:NO)
Then I changed Linker->Advanced->DataExecution Prevention from Image is compatible with DEP (/NXCOMPAT) to Default
And now it works perfectly
Please let me know if this helped you!It'll make me feel better for spending a whole day trying to get it working!
-Nadav
評論2:
The base address randomization seems to benot necessary. Just disable DEP.
大致的意思是說,只需要禁用DEP即可,
在工程的“屬性”->“鏈接器”->“高級”->數據執行保護(DEP),設為“默認”(default)或者“映像與 DEP 不兼容(/NXCOMPAT:NO)“ 即可。(修改后好像不可用)。
注:這個選項只針對Vista有效!!!
問題2:VLD 2.2.3
在項目中使用了visual leak detector,調試時程序無法啟動報錯“應用程序正常啟動失敗(0xc0150002)”。
解決流程:
查看vs輸出信息最后一條是:
Theprogram '[3980] MobileSignalAnalyzer.exe: Native' has exited with code-1072365566 (0xc0150002)
在網上多方查找有:
http://blog.csdn.net/evilswords/article/details/5698851
http://blog.csdn.net/brook0344/article/details/6685724
這兩篇有解決辦法,就是把VLD中的這兩個復制到執行文件夾下就正常了
Microsoft.VC90.CRT.manifest
Microsoft.DTfW.DHL.manifest
產生原因:
VC2003、VC2005、VC2008及其后續版本,對底層最基本的CRT、MFC、ATL庫都進行了重構,為了避免不同版本的庫引起沖突,重構后的庫文件一般放在C://windows/WinSxS 文件夾中,並用特定的文件夾/文件名稱進行標識;
與VC6不同, VC2003、VC2005、VC2008及其后續版本,引入了manifest清單的概念,即應用程序編譯后會同時生成對應的.manifest文件,並將該.manifest文件作為資源編譯到dll或者exe中去。.manifest文件實際上是一個XML格式的文本文件,里面記錄了dll或exe中要引用的CRT、MFC、ATL庫的版本和名稱。VC6編譯的應用程序對CRT、MFC、ATL的dll都是直接調用,而VC2003、VC2005、VC2008編譯的程序都是先查詢編譯到資源中的manifest中的記錄,然后按照記錄提供的版本和名稱去搜尋對應的CRT、MFC、ATL庫以及隨庫發布的.manifest文件,搜尋的路徑包括當前目錄、C://windows/WinSxS等等,如果沒有找到對應的庫文件,則提示“應用程序正常初始化失敗”。