Spring4D中的TForm管理

时间:2016-07-18 21:43:55

标签: delphi spring4d

我有以下代码:

Project.dpr

program Project2;

uses
  madExcept,
  madLinkDisAsm,
  madListHardware,
  madListProcesses,
  madListModules,
  Spring.Container,
  Vcl.Forms,
  uRegistrations in '..\Memory leak II\uRegistrations.pas',
  Unit3 in 'Unit3.pas' {MainForm},
  Unit4 in 'Unit4.pas' {SecondaryForm},
  Unit5 in 'Unit5.pas';

{$R *.res}

begin
  RegisterTypes(GlobalContainer);
  Application.Initialize;
  Application.MainFormOnTaskbar := True;
//  MainForm:=TMainForm.Create(nil);
  Application.CreateForm(TMainForm, MainForm);
  MainForm.SecondaryForm := Globalcontainer.Resolve<ISecondaryForm>;
  Application.Run;
end.

uRegistrations.pas注册接口

unit uRegistrations;

interface

uses
  Spring.Container;

procedure RegisterTypes(Container: TContainer);

implementation

uses
  Unit5,
  Unit4;

procedure RegisterTypes(Container: TContainer);
begin
  container.RegisterType<ISecondaryForm, TSecondaryForm>.DelegateTo(
    function: TSecondaryForm
    begin
      result := TSecondaryForm.Create(nil);
    end);
  Container.Build;

end;

end.

Unit3.pas持有主要表格

unit Unit3;

interface

uses
  Winapi.Windows,
  Winapi.Messages,
  System.SysUtils,
  System.Variants,
  System.Classes,
  Vcl.Graphics,
  Unit5,
  Vcl.Controls,
  Vcl.Forms,
  Vcl.Dialogs;

type
  TMainForm = class(TForm)
  private
    { Private declarations }
    FSecondaryForm: ISecondaryForm;
  public
    { Public declarations }
    property SecondaryForm: ISecondaryForm read FSecondaryForm write FSecondaryForm;
  end;

var
  MainForm: TMainForm;

implementation

{$R *.dfm}

end.

Unit4.pas与次要表格

unit Unit4;

interface

uses
  Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,
  unit5,
  Vcl.Controls, Vcl.Forms, Vcl.Dialogs;

type
  TSecondaryForm = class(TForm, ISecondaryForm)
  private
    { Private declarations }
  public
    { Public declarations }
  end;

//var
//  SecondaryForm: TSecondaryForm;

implementation

{$R *.dfm}

end.

最后是Unit5.pas和Interface声明

{$M+}
unit Unit5;

interface

type
ISecondaryForm=interface
  ['{62D63E9A-A3AD-435B-8938-9528E70D78B1}']
end;

implementation

end.

它会定期编译并运行,但是当我关闭应用程序时,我有三次内存泄漏。

  

分配号码:8482程序运行时间:721 ms类型:刷柄   句柄:$ 461027f5样式:BS_SOLID颜色:$ f0f0f0

     

分配号码:8318程序运行时间:697 ms类型:TSecondaryForm   地址:$ d51ac64大小:924访问权限:读/写

     

分配号码:8267程序运行时间:693 ms类型:字体句柄   手柄:$ 1d0a28f1脸:Tahoma身高:-11

为什么会发生这种情况,我该如何解决?

修改

在回答之后,我实施了以下解决方案(评论突出了我得到的错误:

procedure RegisterTypes(Container: TContainer);
begin
  container.RegisterType<ISecondaryForm, TSecondaryForm>.DelegateTo(
    function: TSecondaryForm
    begin
      result := TSecondaryForm.Create(nil);

      result.Owner:=Application.MainForm;//cannot assign to a read-only property
      result.Parent:=Application; //incompatible types
      result.Parent:=application.MainForm;//memory leak

    end);
  Container.Build;

end;

我还尝试以下列方式修改TSecondaryForm的OnClose方法:

procedure TSecondaryForm.FormClose(Sender: TObject; var Action: TCloseAction);
begin
  Action:=caFree; //memory leak
end;

但是我有内存泄漏。

上述所有技术我做错了什么?

最后,我只是按照评论中的建议设置了两个方法 _AddRef _Release 来管理引用计数,我没有更多的内存泄漏。

  TSecondaryForm = class(TForm, ISecondaryForm)
  private
    { Private declarations }
  protected
    FRefCount: Integer;
    function _AddRef: Integer; stdcall;
    function _Release: Integer; stdcall;
  public
    { Public declarations }
  end;

function TSecondaryForm._AddRef: Integer;
begin
  Result := InterlockedIncrement(FRefCount);
end;

function TSecondaryForm._Release: Integer;
begin
  Result := InterlockedDecrement(FRefCount);
  if Result=0 then
    self.Free;
end

2 个答案:

答案 0 :(得分:5)

如果您希望通过接口引用计数处理表单(或从TComponent继承的任何类),那么您需要自己实现它(请查看System.TInterfacedObject作为如何执行此操作的示例)。< / p>

您基本上需要重新实现IInterface到要启用引用计数的类:

type
  TInterfacedForm = class(TForm, IInterface)
    // look at System.TInterfacedObject
  end;

如果您这样做,请记住,它不应由所有者机制处理。如果您将其注册到容器并使用其默认创建机制,则从Spring4D 1.2开始将nil传递给所有者 - 请参阅Spring.Container.Resolvers.TComponentOwnerResolver)。在任何版本中,您需要在DelegateTo内使用nil显式创建它。

如果您正在处理通过其父属性放置到其他控件(如框架)上的接口的任何控件,请记住,在这种情况下,另一个内存管理机制会发挥作用,如果其父级可能会破坏此类组件正在被破坏 - 如果你只是处理不是问题的接口形式,但我想我在这里提到完整性。

答案 1 :(得分:1)

TComponent后代(如TForm)禁用接口引用计数,因此没有人释放辅助表单。内存模型是基于所有者的,也就是说,当拥有一个对象的父对象被释放时,它会释放所有它的孩子。

因此,您可以将所有者传递给工厂函数上的表单(可能是ApplicationApplication.MainForm)并遵守TComponent的内存模型或在{上添加一个钩子{1}}表单事件,并将OnClose设置为Action。前者将在申请被关闭时销毁表格,后者将在二级表格关闭后尽快销毁(