FastMM 定位内存泄露的代码位置(转)

FastMM 定位内存泄露的代码位置

开源的FastMM,使用很简单,在工程的第一行引用FastMM4即可(注意,一定要在第一个Uses的位置),可以在调试程序时提示内存泄露情况,还可以生成报告。

在Delphi2007以后版本中,使用更加简单,只需要在工程开始的位置加上语句:

ReportMemoryLeaksOnShutdown := True;就可以了,并且在运行时不会出现提示。如果想要生成文件报告,还需要FastMM4,Delphi中没有别的设置可以生成文件报告。

可以修改FastMM4Options.inc中的参数开关来修改内存管理的相关设置。

使用说明如下:

1.打开FastMM4的调试功能

首先在自己的project里把FastMM4放在最前面,例如:

FastMM4,

Main   in   ‘Main.pas’   {MainForm},

再修改FastMM4Options.inc,打开全调试模式。例:

{$define   FullDebugMode}

也可以在project中定义编译常量:FullDebugMode。

同时把FastMM_FullDebugMode.dll拷贝到编译后生成的可执行程序所在目录。

再要打开内存泄漏报告:EnableMemoryLeakReporting。一般情况下是缺省打开的。  这样就打开了全调试模式,如果发生内存泄漏将会生成报告文件,如果在IDE运行的时候还会弹出一个对话框显示。报告文件类似:XXX_MemoryManager_EventLog.txt

2.报告文件由两部分组成,并且是每次运行append。

第一部分是泄漏的详细内容,将每个没释放的内存块详细信息显示出来。例:

A   memory   block   has   been   leaked.   The   size   is:   28

{一个28字节的内存块在程序结束后没有被释放}

{这个内存块在分配的时候的调用堆栈,也就是Call   Stack,可以清楚看出调用函数的次序。如果是系统dll则还有相应的函数名。}

Stack   trace   of   when   this   block   was   allocated   (return   addresses):   
4028E7

4030EC

406649

412365

41236E

411DD3

426B45

427236

42888C

{这个内存类型,如果是字符串string或TObject继承的对象则会显示名称及行号。}

The   block   is   currently   used   for   an   object   of   class:   Unknown

{将内存块头256个字符显示出现,作为内容提示。}

Current   memory   dump   of   256   bytes   starting   at   pointer   address   107BDD8:

第二部分是总结性内容,例:

{这个小型内存块泄漏的报告,如果有大型内存块泄漏则会加一行专门提示大型内存块泄漏。}This   application   has   leaked   memory.   The   small   block   leaks   are   (excluding   expected   leaks   registered   by   pointer):

{21-28字节的内存块泄漏,未知类型一个}

21   -   28   bytes:   Unknown   x   1

Note:   Memory   leak   detail   is   logged   to   a   text   file   in   the   same   folder   as   this   application.   To   disable   this   memory   leak   check,   undefine   “EnableMemoryLeakReporting”.

有了这份报告只不过了解到内存泄漏存在,但是哪里没释放就还需要更进一步地调查。

调查的目标有:

  1. 内存块分配在哪个函数里哪段代码。

这个在报告里可以结合内容和调用堆栈来看。前256个字节可以进行分析,推测分配者,调用堆栈就直接指出了分配函数,不过是一些地址,不能直接知道函数名和代码段。这时候就需要在delphi   ide环境下查看二进制内存映像了,就是View   CPU功能。在设定断点并停下后,可以View   CPU,在菜单View=>Debug   Window=>CPU ,快捷键:Ctrl+Alt+C,View CPU窗口正中就是内存映像,而且源码也相应地标注好了,左边列的地址就是内存报告中的Call   Stack中的地址,翻页找到所对应的代码就知道哪里分配内存了。

2.检查释放内存的地方是否被调用

可以用日志或断点来调试,如果压根就没有释放内存那就补上代码,如果有却没有执行则检查一下执行条件是否正确,如果断点没起作用很可能是因为代码永远不会被执行(死代码)。这要靠经验和调试,基本上借助IDE和内存报告就可以很好地防止内存泄漏。同时要加强测试用例,争取在测试用例中能遍历到所有的代码和大部分关键功能,这样内存泄漏报告就会更准确一点。

fastmm每次在程序关闭后就会根据情况生成内存泄漏报告,如果没有弹出内存泄漏警告则恭喜你,内存管得很好。

另外:

1).内存管理不是GC自动回收内存,而是检查是否有泄漏。   
  2).windows系统的内存泄漏是无法检查的,仅限于应用程序内部,不过检查出系统泄漏也没办法,只能等更新了。       
  3).检查泄漏后要自己去检查代码补齐内存释放,报告并不能做这事

时间: 2024-09-30 22:46:34

FastMM 定位内存泄露的代码位置(转)的相关文章

[VS] 使用Visual Studio查找和定位内存泄露 @Windows

一,代码示例 1 #include <stdio.h> 2 3 void* memleak1(); 4 void* memleak2(); 5 6 int main() 7 { 8 void *p1 = memleak1(); 9 void *p2 = memleak2(); 10 11 printf("p1=%p, p2=%p\n", p1, p2); 12 13 return 0; 14 } main.c 1 /* memleak1.c */ 2 #include &l

利用linux的mtrace命令定位内存泄露(Memory Leak)

一谈到内存泄露, 多数程序猿都闻之色变. 没错, 内存泄露非常easy引入. 但非常难定位.  以你我的手机为例(如果不常常关机). 如果每天泄露一些内存, 那么開始的一个星期, 你会发现手机好好的. 当内存泄露积累到一定程度,  那就是各种卡死了. 系统异常, 最后死机. 不得不重新启动. 假设搞开发. 遇到内存泄露问题, 那就呵呵了. 你可能先得花好几天来复现问题(泄露积累), 然后须要花好几天来定位问题和改动问题, 然后又要花好几天来验证问题, 并且. 非常有可能没法一次改好, 上述流程又

使用valgrind定位内存泄露

简单备忘一下,valgrind的用法: valgrind --tool=memcheck --leak-check=full --log-file=a.log ./a.exe 其中: --tool指定工具: --leak-check指定检测级别: --log-file指定输出到文件,不给出这个字段则输出到屏幕打印. valgrind的错误含义:(转自http://blog.csdn.net/ai2000ai/article/details/51131645) Conditional jump o

Bullet(Cocos2dx)之内存泄露检测

编写的程序难免会有内存泄露,为了检测内存泄露,可以采取各种各样的措施, 今天向大家介绍一款windows下的内存检测工具Visual Leak Detector for Visual C++ 去官网下载最新版本安装. 用VS建立一个C++控制台项目, 1.将vld安装目录下的include的头文件copy到c++目录下 2.把lib/win32(如果vs版本为64的,则选择win64)copy到c++目录, //3.把bin/win32copy到c++的Debug目录(先Debug运行c++)下

一种内存泄露检查和定位的方法

一个系统后台服务进程,可能包括多个线程,在生成环境下要求系统程序能够稳定长时间稳定运行而不宕机.其中一个基本的前提就是需要保证系统程序不存在内存泄露.那么,该如何判读系统程序是否存在内存泄露呢?如果存在,又该如何检测呢? 0.判读系统程序是否存在内存泄露 对于频繁快速申请内存的应用,可以允许下面的命令: top -p `pidof YourProgrogram` 如果看到系统内存使用率一直上身,没有下降,就说明很可能存在内存泄露. 对于申请.使用内存比较缓慢的应用程序,可以通过下面的命令,观测一

代码内存泄露检测(1) MLeaksFinder (Wechat开源) + FBRetainCycleDetector (FaceBook开源)

每次项目编译完成之后,都被内存搞得头昏脑胀,压力甚大. 利用两周时间,稍微研究了 微信开源的 MLeaksFinder 和 facebook 开源的 FBMemoryProfiler, 这两个开源三方,在编写过程中就可以检测内存泄露,实在是不要太方便-- 希望自己在下一个项目能用的得心应手-- 1.  微信 MLeaksFiner 如果对它,你还是不是很了解 ,可以到查阅这里:  http://wereadteam.github.io/2016/07/20/MLeaksFinder2/ ; 1.

Android 内存泄露总结(附内存检测工具)

https://segmentfault.com/a/1190000006852540 主要是分三块: 静态储存区:编译时就分配好,在程序整个运行期间都存在.它主要存放静态数据和常量. 栈区:当方法执行时,会在栈区内存中创建方法体内部的局部变量,方法结束后自动释放内存. 堆区:通常存放 new 出来的对象.由 Java 垃圾回收器回收. 栈与堆的区别 栈内存用来存放局部变量和函数参数等.它是先进后出的队列,进出一一对应,不产生碎片,运行效率稳定高.当超过变量的作用域后,该变量也就无效了,分配给它

内存泄露、内存溢出以及解决方法

内存泄露是指程序在运行过程中动态申请的内存空间不再使用后没有及时释放,从而很可能导致应用程序内存无线增长.更广义的内存泄露包括未对系统的资源的及时释放,比如句柄等. 内存溢出即用户在对其数据缓冲区操作时,超过了其缓冲区的边界:尤其是对缓冲区写操作时,缓冲区的溢出很可能导致程序的异常. 一.内存泄露 "知己知彼,方能百战不殆",如果我们能够比较清楚的了解在编程的时候哪些情况容易导致内存泄露,通过避免这些糟糕的情况,从提高代码的质量本身出发,来抵御潜在导致内存泄露的发生. 1.1先来看看内

Visual Studio(或者VC)内存泄露工具

使用简介 在写程序的过程中,难免会遇到内存泄露的时候,这个时候如果手工查找内存泄露,不说方法没有通用的,就是真的要自己手工查找也是很耗时间和精力的.诚然,我们可以借助一些工具,而且我们还会惊奇地发现这些工具很有用(比如Intel的内存泄露检测工具),但是因为往往这些工具安装比较麻烦,而我们写的程序又不是很大,所以我们或许可以找个更小巧的方法.微软就提供了这样的方法,我们只需要在程序中加入几行代码,就可以发现内存泄露的问题,然后我们就可以定位内存泄露了(自己用几行代码就可以实现,神奇!).那么如何