【发布时间】:2012-02-05 19:01:04
【问题描述】:
我正在尝试检测是否有一块内存没有被释放。当然,经理通过对话框或日志文件告诉我,但如果我想将结果存储在数据库中怎么办?例如,我想在数据库表中有一个分配给定块的例程名称。
在阅读了 FastMM 的文档后,我知道从 4.98 版开始,我们可能会在发生内存分配、释放和重新分配时收到管理器的通知。例如OnDebugFreeMemFinish 事件向我们传递了一个包含有用信息的PFullDebugBlockHeader。
PFullDebugBlockHeader 缺少一件事 - 如果给定块已被应用程序释放的信息。
除非OnDebugFreeMemFinish 只为未释放的块调用?这是我不知道并想知道的。
问题是,即使连接到OnDebugFreeMemFinish 事件,我也无法确定该块是否被释放。
这是一个例子:
program MemLeakTest;
{$APPTYPE CONSOLE}
uses
FastMM4, ExceptionLog, SysUtils;
procedure MemFreeEvent(APHeaderFreedBlock: PFullDebugBlockHeader; AResult: Integer);
begin
//This is executed at the end, but how should I know that this block should be freed
//by application? Unless this is executed ONLY for not freed blocks.
end;
procedure Leak;
var
MyObject: TObject;
begin
MyObject := TObject.Create;
end;
begin
OnDebugFreeMemFinish := MemFreeEvent;
Leak;
end.
我缺少的是这样的回调:
procedure OnMemoryLeak(APointer: PFullDebugBlockHeader);
浏览了FastMM的源码后看到有一个程序:
procedure LogMemoryLeakOrAllocatedBlock(APointer: PFullDebugBlockHeader; IsALeak: Boolean);
这可以被覆盖,但也许有更简单的方法?
【问题讨论】:
-
我一直明白 FastMM 只能将此检查作为程序应该执行的最后一个操作 - 根据定义 - 所以当 FastMM 报告您的代码时,您的代码已经完成。要获得部分解决方案,您可以随时查看他们的源代码,了解分配的内存是如何标记的。
-
按预期泄漏?您是否按预期注册了它。此外,除非您提供理解预期寿命的复杂逻辑,否则您无法确定内存是否泄漏,直到关闭。
-
如果
OnDebugFreeMemFinish被调用,这意味着块被释放。没有OnMemoryLeak事件。永远不可能有这样的事件。 FastMM 所做的是,在关闭时,确定任何尚未释放的块都必须是泄漏的。它无法在此之前检测到泄漏。 -
每当 FastMM 告诉我存在内存泄漏时,我都会关闭工具并立即修复它。如果你不这样做,那么你会发现很难重现泄漏。如果您真的希望登录到数据库,那么您需要查看 CheckBlocksOnShutdown 函数。另一个潜在的扩展点是
AppendEventLog,但你需要修改我怀疑的 FastMM 源。 -
Erm 只是拿起文件,解析它并把它放在数据库中?
标签: delphi memory-leaks delphi-2009 fastmm