如何正确地释放Delphi表单

时间:2017-10-03 08:01:24

标签: delphi fastmm

尝试使用FastMM在完全调试模式下修复我的应用程序中的内存泄漏,报告了一些关于TForm(后代)实例的泄漏。检查并单步执行代码后,我可以肯定地说这些表单是已发布而不是已释放。看起来FastMM正在检查泄漏,然后才会发布Release发布的实际释放。

MyForm: TForm;
MyForm := TForm.Create(nil);
...
MyForm.Release; // FastMM reports MyForm as a leak

我尝试释放,而不再报告这些内存泄漏,但有时会在释放时发生访问冲突:

MyForm: TForm;
MyForm := TForm.Create(nil);
...
MyForm.Free; // // FastMM does not report MyForm as a leak, but sometimes an access violation is triggered

我尝试释放,然后立即调用Application.ProcessMessages,据我所知,可以在https://stackoverflow.com/a/917187/1465896中阅读ZoëPeterson的回答。我必须理解错误,因为它总是在Application.ProcessMessages中立即崩溃:

MyForm: TForm;
MyForm := TForm.Create(nil);
...
MyForm.Release; 
Application.ProcessMessages; // always triggers an access violation 

我不希望将这些表单注册为预期的内存泄漏,因为它们包含更多混乱内存泄漏日志文件的对象,从而更难找到更重要的泄漏。

所以我的问题是,如何正确地释放Delphi表单以便FastMM不会将其报告为泄漏?

根据要求,这是一个SSCnCE(简短,自包含,正确(不可编译,因为我不知道如何发布项目),例子):

File FormRelease.dpr:

program FormRelease;

uses
  FastMM4 in 'FastMM4.pas',
  Vcl.Forms,
  MainForm_fm in 'MainForm_fm.pas' {MainForm},
  MyForm_fm in 'MyForm_fm.pas' {MyForm};

{$R *.res}

begin
  Application.Initialize;
  Application.MainFormOnTaskbar := True;
  Application.CreateForm(TMainForm, MainForm);
  Application.Run;
end.

文件MainForm_fm.pas:

unit MainForm_fm;

interface

uses
  Vcl.Forms,
  MyForm_fm;

type
  TMainForm = class(TForm)
    procedure FormCreate(Sender: TObject);
    procedure FormDestroy(Sender: TObject);

  private
    FMyForm: TMyForm;
  end;

var
  MainForm: TMainForm;

implementation

{$R *.dfm}

procedure TMainForm.FormCreate(Sender: TObject);
begin
  FMyForm := TMyForm.Create(nil);
end;

procedure TMainForm.FormDestroy(Sender: TObject);
begin
  FMyForm.Release;
end;

end.

档案MyForm_fm.pas:

unit MyForm_fm;

interface

uses
  Vcl.Forms;

type
  TMyForm = class(TForm)
  end;

implementation

{$R *.dfm}

end.

摘自文件FormRelease_MemoryManager_EventLog.txt,启动程序并使用alt-F4停止它:

This application has leaked memory. The small block leaks are (excluding expected leaks registered by pointer):

13 - 20 bytes: TList x 1, Unknown x 1
21 - 36 bytes: TPen x 1, TMargins x 1, TPadding x 1, TIconImage x 1, TBrush x 2, TTouchManager x 1, TSizeConstraints x 1, UnicodeString x 1, Unknown x 3
37 - 52 bytes: TGlassFrame x 1, TFont x 2
53 - 68 bytes: TIcon x 1
69 - 84 bytes: TControlScrollBar x 2
101 - 116 bytes: TControlCanvas x 1
149 - 164 bytes: Unknown x 1
917 - 1012 bytes: TMyForm x 1

如果TMyForm x 1张贴CM_RELEASE已被处理,为什么会留下FMyForm.Release

1 个答案:

答案 0 :(得分:1)

这听起来像是两件事。

首先,您明确地创建了表单,因此您应该明确地释放它。除了特殊情况之外,您不应该调用Release,除非是从一个自己的事件处理程序中释放自己的表单。

MyForm: TForm;
...
MyForm := TForm.Create(nil);
try
  // Only reference MyForm in this block
finally
  MyForm.Free;
end;

其次,调用Free并不会导致问题。我认为它揭示了它。如果我不得不猜测,则会发生访问冲突,因为MyForm或其中一个组件正在此块之外被引用。使用调试器查找此引用并进行修复。