未破坏未使用的接口引用

时间:2011-01-13 14:44:08

标签: delphi interface delphi-2010

我的应用程序中有另外一个由于粗心使用Delphi接口而导致的错误。当我将接口传递给忽略该参数的过程时,实例永远不会被释放。请参阅以下简单示例:

ITest = interface
    procedure Test;
end;

Tester = class(TInterfacedObject, ITest)
public
    procedure Test;
end;

Base = class
public
    procedure UseTestOrNot(test : ITest); virtual; abstract;
end;

A = class(Base)
public
    procedure UseTestOrNot(test : ITest); override;
end;

B = class(Base)
public
    procedure UseTestOrNot(test : ITest); override;
end;

{ A }

procedure A.UseTestOrNot(test: ITest);
begin
    test.Test();
end;

{ B }

procedure B.UseTestOrNot(test: ITest);
begin
    WriteLn('No test here');
end;

// -------- Test ---------------------------------------
var
    list : TObjectList<Base>;
    x : Base;
    t : ITest;
begin
    ReportMemoryLeaksOnShutdown := true;

    list := TObjectList<Base>.Create;
    list.Add(A.Create);
    list.Add(B.Create);

    // 1 x Tester leak for each B in list:
    for x in list do
        x.UseTestOrNot(Tester.Create);

    // this is ok
    for x in list do
    begin
        t := Tester.Create;
        x.UseTestOrNot(t);
    end;

    list.Free;
end.

您能否解释一下参考计数器出了什么问题? 你能给出任何最佳实践/指南(比如“永远不要在函数调用中创建一个接口实例[如果你不知道里面发生了什么]。)

我能想到的最佳解决方案是在类Base中编写一个模板方法,用于保存传递的测试实例并调用抽象的DoUseTestOrNot方法。

修改 Delphi 2010

2 个答案:

答案 0 :(得分:8)

为你添加一个guid ITest声明

ITest = interface
['{DB6637F9-FAD3-4765-9EC1-0A374AAC7469}']
    procedure Test;
end;

将循环更改为此

for x in list do
    x.UseTestOrNot(Tester.Create as ITest);

GUID必须能够使用as

Test.Create as ITest使编译器在创建的对象超出范围的位置添加发布。

答案 1 :(得分:8)

<击> 这是错误here的不同表现形式 我将把它添加到质量控制报告中。

这不再在Delphi XE update 1中重现。

- 的Jeroen