Miałem kolejny błąd w mojej aplikacji spowodowany przez nieostrożne korzystanie z interfejsów Delphi. Kiedy przekazuję interfejs do procedury, która ignoruje ten argument, instancja nigdy nie jest zwalniana. Zobacz następujący prosty przykład:Nieużywany odnośnik interfejsu nie został zniszczony.
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.
Czy możesz wyjaśnić, co jest nie tak z licznikiem referencyjnym? Czy możesz podać najlepszą praktykę/wytyczne (np. "Nigdy nie twórz instancji z interfejsem wewnątrz wywołania funkcji [jeśli nie wiesz, co dzieje się w środku].)
Najlepsze rozwiązanie, jakie mogę wymyślić dla tego przykładu, to napisać metodę szablonu w klasie bazowej, która zbawia przekazany egzemplarz testowy i wywołuje abstrakcyjną DoUseTestOrNot
metody.
EDIT Delphi 2010
Jaka jest wersja Delphi? –
Niestety, to Delphi 2010 – hansmaad
Nie ma znaczenia, jest to błąd niezależnie od wersji. czekać do delphi ex 2? Pozdrawiam: – APZ28