我的应用程序中有另外一个由于粗心使用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
答案 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)