最近将一个指纹匹配算法按照AFIS系统的接口标准做成dll上服务器测试,结果在进行200万人指纹的大库比对时出错了,通过分析发现是dll中存在内 存泄露导致系统资源耗尽。虽然一次只有那么200kByte,但乘上200万这个基数也是不小。于是找来了两个工具来检测代码中的内存泄露:VLD和 LeakDiag。
1.Visual Leak Detector 1.0 (VLD) mhfx.yaowan.com 这是一个小巧内存检测工具,是为Visual C++用户设计的。其特色为: 简单易用,只需要在需要进行内存泄漏检测的的主程序代码中加入#include "VLD.h“,并将编译好的Lib文件放入默认的库目录,然后在Visual C++用Debug模式编译运行程序,程序运行结束后便可在Visual C++的输出窗口看到内存泄露检测结果。 通过宏定义控制测试行为,如报告的详略等。 支持Windows X64。 检测报告很直观,甚至能显示泄漏内存的内容。 下面是一段用于演示VLD功能的代码: TestVLD.cpp #include "stdafx.h" #include "vld.h" #include "afx.h" #include "string.h" typedef void(*FARPROC2)(int, char*); // 用于调用dll中函数的指针 void leak5Bytes(); // 泄漏5个字节内容为"ABCDE"内存 void leakInDll(); // 调用dll中会造成内存泄漏函数 int _tmain(int argc, _TCHAR* argv[]) { int * p = new int[10]; for (int i=0; i<10; ++i) p[i]=i; // 泄漏10个int的内存,内容分别为0到9 leak5Bytes(); // 泄漏5个字节内存 leakInDll(); // 调用dll中会造成内存泄泄漏的函数 return 0; } void leak5Bytes() { char * p = new char[5]; for (int i=0; i<5; ++i) p[i]='A'+i; // 泄漏的5个字节内存为"ABCDE" } void leakInDll() { char *p = new char[16]; strcpy(p, "dll leaking!"); // 调用dll函数,在dll中分配一个16字节的空间保存"dll leaking!",且不释放内存。 HMODULE hMod = LoadLibrary(_T("Dll.dll")); if (hMod) { FARPROC2 hLeakInDll = (FARPROC2) GetProcAddress(hMod, "dllLeak"); hLeakInDll(16, p); } } 下面是Dll中被调用的函数代码 void dllLeak(int n, char * st) { char *p = new char[n]; for (int i=0; i<n; ++i) p[i] = st[i]; } 运行完毕之后的输出为: WARNING: Visual Leak Detector detected memory leaks! ---------- Block 57 at 0x01C899D8: 16 bytes ---------- Call Stack: f:\rtm\vctools\crt_bld\self_x86\crt\src\new2.cpp (27): operator new[] f:\rtm\vctools\crt_bld\self_x86\crt\src\crt0.c (318): __tmainCRTStartup f:\rtm\vctools\crt_bld\self_x86\crt\src\crt0.c (187): wmainCRTStartup 0x7D517D2A (File and line number not available): BaseProcessInitPostImport Data: 64 6C 6C 20 6C 65 61 6B 69 6E 67 21 00 CD CD CD dll.leak ing!.... ---------- Block 56 at 0x01C89990: 5 bytes ---------- Call Stack: f:\rtm\vctools\crt_bld\self_x86\crt\src\new2.cpp (27): operator new[] f:\rtm\vctools\crt_bld\self_x86\crt\src\crt0.c (318): __tmainCRTStartup f:\rtm\vctools\crt_bld\self_x86\crt\src\crt0.c (187): wmainCRTStartup 0x7D517D2A (File and line number not available): BaseProcessInitPostImport Data: 41 42 43 44 45 ABCDE... ........ ---------- Block 55 at 0x01C89928: 40 bytes ---------- Call Stack: f:\rtm\vctools\crt_bld\self_x86\crt\src\new2.cpp (27): operator new[] f:\rtm\vctools\crt_bld\self_x86\crt\src\crt0.c (318): __tmainCRTStartup f:\rtm\vctools\crt_bld\self_x86\crt\src\crt0.c (187): wmainCRTStartup 0x7D517D2A (File and line number not available): BaseProcessInitPostImport Data: 00 00 00 00 01 00 00 00 02 00 00 00 03 00 00 00 ........ ........ 04 00 00 00 05 00 00 00 06 00 00 00 07 00 00 00 ........ ........ 08 00 00 00 09 00 00 00 ........ ........ Visual Leak Detector detected 3 memory leaks. 可以看到,VLD检测出了分别在主程序函数t_main、leak5Bytes以及Dll.dll中dllLeak的内存泄漏,并显示了泄漏内存的内容。 2. LeakDiag 之前的VLD必须和源代码一起编译,若是遇到主程序源码不方便重新编译的情况,则可以选择使用LeakDiag。LeakDiag是微软的一款内存泄漏检测工具,鉴于已经有大牛写了比较详尽的使用说明,我就不再赘述,直接引用大牛的博客地址: http://www.cppblog.com/sandy/archive/2008/08/18/59260.html long.yaowan.com LeakDiag下载地址: ftp://ftp.microsoft.com/PSS/Tools/Developer%20Support%20Tools/LeakDiag/ 工具: 1. IBM Rational PurifyPlus是一套完整的运行时分析工具,旨在提高应用程序的可靠性和性能。PurifyPlus将内存错误和泄漏检测、应用程序性能描述、代码覆盖分析等功能组合在一个单一、完整的工具包中。 网站: http://www-128.ibm.com/developerworks/rational/products/purifyplus 从这里可以得到 IBM Rational 的试用版(DVD), dp.peiyou.com http://www-128.ibm.com/developerworks/cn/offers/sek/index.html 2. BoundsChecker BoundsChecker是一个C++运行时错误检测和调试工具。它通过在VisualStudio内自动化调试过程加速开发并且缩短上市的周期。BoundsChecker提供清楚,详细的程序错误分析,许多是对C++独有的并且在 static,stack和heap内存中检测和诊断错误,以及发现内存和资源的泄漏。3. http://www.parasoft.com/ 4. http://www.glowcode.com/