教你如何实现内存泄漏检查

开发 开发工具
关于内存泄漏的检查网上有很多的例子和代码,今天我们将一步一步教你如何实现内存泄漏检查。

  关于内存泄漏的检查网上有很多的例子和代码,其基本的方法都是用宏,替换掉内存分配以及释放的函数。但是现在网上很多的例子中没有一个是适合我们公司的需求的。

  具体的对内存泄漏检查有如下要求:

  1. 内存泄漏检查的代码尽可能少的占用CPU及内存

  2. 尽可能的不影响原程序

  因为,我们的服务器程序有泄漏而且是特殊情况下会泄漏,平时很难模拟出来。

  对于这种情况下的内存泄漏我以前的做法如下:

  1. 用写文件的方法记录所有的内存分配以及释放的操作

  2. 再写一个工具去分析所有的记录,从中找出泄漏的代码

  这样做需要大量的硬盘空间,不过,这个无所谓了现在硬盘很便宜!

  不过需要考虑到服务器程序当中包含了exe以及多个dll,为了通用,内存泄漏检查分为下面几个部分:

  1. IMemLeak.h IMemLeak.cpp 加入每一个模块当中

  2. MemLeakLog.dll 统一记录所有的内存操作,将其记录到文件当中

  3. MemCheckTool.exe 分析工具

  1. //IMemLeak.h  
  2.   #ifndef _YG_MEMDBG_H_  
  3.   #define_YG_MEMDBG_H_  
  4.   #include <cstdlib> 
  5.   //Redefines  
  6.   #definemalloc(size) mallocb(size, __FILE__, __LINE__)  
  7.   #definefree(memblock) freeb(memblock, __FILE__, __LINE__)  
  8.   #definerealloc(memblock, size) reallocb(memblock, size, __FILE__, __LINE__)  
  9.   #definecalloc(num, size) callocb(num, size, __FILE__, __LINE__)  
  10.   //Redefined functions  
  11.   void* mallocb(size_t size, constchar*pszFile, intnLine);  
  12.   voidfreeb(void*memblock, constchar*pszFile, intnLine);  
  13.   void* reallocb(void*memblock, size_t size, constchar*pszFile, intnLine);  
  14.   void* callocb(size_t num, size_t size, constchar*pszFile, intnLine);  
  15.   //For C++  
  16.   void* operatornew(size_t size, constchar*pszFile, intnLine);  
  17.   void* operatornew[](size_t size, constchar*pszFile, intnLine);  
  18.   voidoperatordelete(void*pvMem) throw();  
  19.   voidoperatordelete[](void*pvMem) throw();  
  20.   voidpre_delete(constchar*pszFile, intnLine);  
  21.   //Redefine new and delete  
  22.   #definenewnew(__FILE__, __LINE__)  
  23.   #definedelete pre_delete(__FILE__, __LINE__),delete  
  24.   #endif  
  25.   //IMemLeak.cpp  
  26.   #include <stdio.h> 
  27.   #include <tchar.h> 
  28.   #include <stdlib.h> 
  29.   #include <malloc.h> 
  30.   #include <Windows.h> 
  31.   #include <cstdlib> 
  32.   enumEOperateType  
  33.   {  
  34.   Type_Malloc,  
  35.   Type_Calloc,  
  36.   Type_Realloc,  
  37.   Type_New,  
  38.   Type_New_Array,  
  39.   Type_Free,  
  40.   Type_Delete,  
  41.   Type_Delete_Array  
  42.   };  
  43.   typedef void(__stdcall * pFun_MemLeakLog)(LPCSTR PLog);  
  44.   pFun_MemLeakLog MemLeakLog = NULL;  
  45.   voidCheckMemLeakLogDLL()  
  46.   {  
  47.   if(MemLeakLog == NULL)  
  48.   {  
  49.   HINSTANCE hinstLib = LoadLibrary(_T("MemLeakLog.dll"));  
  50.   if(hinstLib != NULL)  
  51.   {  
  52.   MemLeakLog = (pFun_MemLeakLog)GetProcAddress(hinstLib, "MemLeakLog");  
  53.   }  
  54.   }  
  55.   }  
  56.   voidLog(EOperateType type, void* pmem, size_t size, intnLine, constchar* pszFile)  
  57.   {  
  58.   CheckMemLeakLogDLL();  
  59.   chartemp[1024];  
  60.   if(MemLeakLog != NULL)  
  61.   {  
  62.   memset(temp, 0, 1024);  
  63.   sprintf_s(temp, 1024, "%d-%p-%d-%d [%s] ", type, pmem, size, nLine, pszFile);  
  64.   MemLeakLog(temp);  
  65.   }  
  66.   }  
  67.   void* mallocb(size_t size, constchar*pszFile, intnLine)  
  68.   {  
  69.   void* pRet = malloc(size);  
  70.   Log(Type_Malloc, pRet, size, nLine, pszFile);  
  71.   returnpRet;  
  72.   }  
  73.   void* callocb(size_t num, size_t size, constchar*pszFile, intnLine)  
  74.   {  
  75.   void* pRet = calloc(num, size);  
  76.   Log(Type_Calloc, pRet, size, nLine, pszFile);  
  77.   returnpRet;  
  78.   }  
  79.   voidfreeb(void*memblock, constchar*pszFile, intnLine)  
  80.   {  
  81.   if(memblock)  
  82.   {  
  83.   Log(Type_Free, memblock, 0, 0, "NULL");  
  84.   }  
  85.   free(memblock);  
  86.   }  
  87.   void* reallocb(void*memblock, size_t size, constchar*pszFile, intnLine)  
  88.   {  
  89.   void* pRet;  
  90.   pRet = realloc(memblock, size);  
  91.   Log(Type_Free, memblock, size, nLine, pszFile);  
  92.   Log(Type_Realloc, pRet, size, nLine, pszFile);  
  93.   returnpRet;  
  94.   }  
  95.   void* operatornew(size_t size, constchar*pszFile, intnLine)  
  96.   {  
  97.   void* pRet = malloc(size);  
  98.   Log(Type_New, pRet, size, nLine, pszFile);  
  99.   returnpRet;  
  100.   }  
  101.   void* operatornew[](size_t size, constchar*pszFile, intnLine)  
  102.   {  
  103.   void* pRet = malloc(size);  
  104.   Log(Type_New_Array, pRet, size, nLine, pszFile);  
  105.   returnpRet;  
  106.   }  
  107.   //#include <new> 
  108.   voidoperatordelete(void*memblock) throw()  
  109.   {  
  110.   if(memblock)  
  111.   {  
  112.   Log(Type_Delete, memblock, 0, 0, "NULL");  
  113.   }  
  114.   free(memblock);  
  115.   }  
  116.   voidoperatordelete[](void*memblock) throw()  
  117.   {  
  118.   if(memblock)  
  119.   {  
  120.   Log(Type_Delete_Array, memblock, 0, 0, "NULL");  
  121.   }  
  122.   free(memblock);  
  123.   }  
  124.   voidpre_delete(constchar*pszFile, intnLine)  
  125.   {  
  126.   } 

注意:

  a. 输出的目录我是写死了,在D:MemLeak_Log

  b. 在被检查工程里面请增加/FC选项。Project->Properties->Configuration->C/C++->Advanced->Use Full Path Yes(/FC)

  c. MemLeakLog.dll 拷贝到与被检查内存泄漏的进程所在的目录下面

  我附带上一个例子,大家一看就明白了。

  下载地址:http://down.51cto.com/data/236002

原文链接:http://www.cnblogs.com/russinovich/archive/2011/08/12/2135625.html

【编辑推荐】

  1. 发现Java虚拟机内存泄露问题
  2. 再谈让C++更像C#:C#内存泄露问题
  3. 如何发现客户端软件中的内存泄露?
  4. Java内存泄露检测方法揭秘
  5. 揭秘Java内存泄露与溢出的区别

 

责任编辑:彭凡 来源: 博客园
相关推荐

2013-08-02 09:52:14

AndroidApp内存泄漏

2013-12-17 15:46:04

iOS开发iOS 内存泄漏

2024-07-03 11:28:15

2015-04-17 10:35:51

c++c++程序内存泄漏检测代码

2023-12-18 10:45:23

内存泄漏计算机服务器

2021-08-05 15:28:22

JS内存泄漏

2021-08-09 09:54:37

内存泄漏JS 阿里云

2017-09-07 16:52:23

2024-04-19 08:00:00

2011-08-19 14:27:29

iPhone开发

2022-09-15 20:04:14

MemlabJavaScrip内存

2021-03-26 05:59:10

内存检测工具

2022-05-26 09:51:50

JavaScrip内存泄漏

2017-01-05 19:34:06

漏洞nodejs代码

2024-01-30 10:12:00

Java内存泄漏

2021-11-05 08:28:27

内存泄漏调试

2024-03-11 08:22:40

Java内存泄漏

2012-02-22 21:28:58

内存泄漏

2015-03-30 11:18:50

内存管理Android

2023-10-30 08:18:21

内存泄漏Java
点赞
收藏

51CTO技术栈公众号