![]() |
Interfaces: Ich werd' irre!
Hallöle, ich möchte per Interfaces ermöglichen, daß verschiedene Units an ein Objekt rankommen ohne es instantiieren zu müssen (quasi ein Singleton, der von überall nur duch Einbindung der Unit zugreifbar sein soll).
Ich habe es folgendermaßen versucht:
Code:
Das Aufrufen von Test.Show im Hauptprogramm funktioniert auch erstmal gut. Wenn ich die Prozedur allerdings ein zweites Mal ausführe, crasht er mir weg, da vorher anscheinend schon alle Referenzen auf das Interface verschwinden (direkt nach dem ersten Aufruf von Test.Show). Anscheinend besteht das Objekt FTest aber trotzdem noch, da ich per Debugger den RefCount sehen kann.
interface
type ITest = interface ['{74B2F6FC-1464-43E2-A5FB-74A889F723F2}'] procedure Show; end; function Test: ITest; implementation uses Dialogs, SysUtils; type TTest = class(TInterfacedObject, ITest) private public procedure Show; end; var FTest: TTest; function Test: ITest; begin if not Assigned(FTest) then FTest := TTest.Create; Result := FTest as ITest; end; {TTest} destructor TTest.Destroy; begin inherited; end; procedure TTest.Show; begin ShowMessage(''); end; initialization FTest := nil; finalization if Assigned(FTest) then FTest.Free; :wall: Also wird anscheinend nach jedem Zugriff eine Funktion das Objekt FTest freigegeben, was ich eigentlich auch nicht möchte, da die Daten ja eine ganze Weile bereitstehen sollen. Für Anmerkungen und Hilfe wäre ich sehr dankbar. Bitte bringt ein wenig Licht in mein Dunkel. :cry: Bitte Bitte. Fingolfin |
Re: Interfaces: Ich werd' irre!
Ich glaube Du bist auf den einzigen Fehler gestoßen, den Delphi bei Interfaces in Bezug auf die Automatische Referenzzählung besitzt.
Das Problem ist, das Du dir einen Zeiger auf das Objekt merkst, jedoch als Rückgabewert der Funktion einen Interfacezeiger lieferst. Wird der zurückgegebene Interface-Zeiger nicht mehr benötigt, geht der Ref-Counter auf 0 zurück und das Objekt wird freigegeben. Jedoch merkst Du dir den zeiger auf das (nicht mehr vorhandene) Objekt. Und das wird nicht zurückgesetzt. Lösung 1: Nimm als Modulvariable ein ITest statt einem TTest
Delphi-Quellcode:
Lösung 2: Zurücksetzen der Modulvariable im Destruktur
var
FTest: ITest;
Delphi-Quellcode:
Lösung nicht getestet, jedoch aufgrund eigener Erfahrungen müsste das dein Problem lösen.
destructor TTest.Destroy;
begin FTest := nil; inherited; end; |
Re: Interfaces: Ich werd' irre!
:thuimb:
Oha, VIELEN DANK! Jetzt wird mir einiges klar. Und noch besser: Es funktioniert auch noch einwandfrei. Interfaces fielen mir schon immer schwer, aber so langsam lüftet sich der Schleier. :-D Das geht aber echt schnell hier mit hilfreichen Antworten. :) Fingolfin |
Re: Interfaces: Ich werd' irre!
Hallo nochmal. Anscheinend begehe ich immer noch einen ähnlichen Folgefehler.
Ich benutze ITest zur Datenhaltung. Das Interface und dementsprechend auch TTest beinhaltet mehere Collections, die zum Teil selbst Interfaces haben. Die Referenzzählung ist meiner Meinung nach korrekt implementiert, aber in dem Moment, in dem ich Getinterface nutzen möchte, wird der Referenzzähler des jerweiligen Objekt um 1 erhöht und dann wieder gesenkt, was den Aufruf des Destruktors nach sich zieht. Ich möchte eigentlich ein Verhalten, bei dem ich gefahrlos auf ein Interface prüfen kann ohne mit gleich das Objekt zu zerschießen. Vielleicht hast du (oder andere) ja eine Idee, was ich falsch mache. :( Vielen Dank im Voraus, Fingolfin |
Re: Interfaces: Ich werd' irre!
Im Nachhinein glaube ich, mich etwas zu kompliziert ausgedrückt zu habe. Ich versuche es nochmal einfacher:
Hier mein Interface und die Klasse, die es einbindet:
Code:
Dann erstelle ich mein Objekt:
ITest = interface
['{93942268-B484-4BBD-A907-CA8B8B5208BF}'] end; TTest = class(TInterfacedObject, ITest) end;
Code:
Und jetzt möchte ich per GetInterface wissen, ob mein Objekt das ITest interface implementiert:
var
test: TTest; begin test := TTest.Create; end;
Code:
:wall:
var
testint: ITest; begin if test.GetInterface(itest, testint) then begin //hier geht schon nix mehr, weil test freigegeben wurde end; end; Das ist doch ein Grundlagen-Problem oder? Irgendwie fehlt mir der Durchblick. Hilfe :cry: Danke im Voraus, Fingolfin |
Re: Interfaces: Ich werd' irre!
GetInterface - Kenn ich nicht?
Aber dein Problem ist evtl. genau des Bug der Referenzzählung, wenn mit Objekt und Interfacezeigern gemisch operiert wird. Probier aber einfach mal folgendes (ohne Garantie/test):
Delphi-Quellcode:
if test is ITest then
(test as ITest).InterfaceMethode |
Re: Interfaces: Ich werd' irre!
Danke für deinen Kommentar.
Code:
funktioniert ohne Probleme.
(test as ITest).InterfaceMethode
Die Abfrage:
Code:
führt aber leider zu einem "Operator not applicable to this operand type" Fehler. :(
if test is ITest then
@GetInterface Ich war der Meinung, daß das Casten bei Interfaces nur per GetInterface möglich ist. Ohne das is also die Abfrage bringt mir das as leider nichts. :gruebel: Fingolfin |
Re: Interfaces: Ich werd' irre!
Zitat:
dazu könntest du dir die Methode ![]() mfG mirage228 |
Re: Interfaces: Ich werd' irre!
Hallo mirage. Mit
Code:
funktioniert es.
if QueryInterface(ITest, t) = S_OK then
begin t.InterfaceMethode; end; Allerdings steht in der OH ja, daß man gerade QueryInterface vermeiden und GetInterface nutzen soll. GetInterface macht eigentlich auch nur Folgendes:
Code:
Was an der QueryInterface-Variante dann auch noch komisch ist, ist die Tatsache, daß ein danach ausgeführtes
if GetInterface(IID, Obj) then
Result := S_OK else Result := E_NOINTERFACE;
Code:
anscheinden nix bewirkt.
t := nil;
Jetzt bin ich echt verwirrt, soll ich jetzt direkt QueryInterface nehmen weil's funktioniert, obwohl die OH es nicht gut findet? Ich habe da etwas Angst, daß mir das später noch irgendwelche Probleme bereitet. Danke im Voraus, Fingolfin |
Re: Interfaces: Ich werd' irre!
Oha, habe natürlich Unsinn erzählt.
Anstatt:
Code:
muß es natürlich:
if QueryInterface(ITest, t) = S_OK then
begin t.InterfaceMethode; end;
Code:
heißen. :wall: Und das funktioniert genausowenig wie QueryInterface.
if test.QueryInterface(ITest, t) = S_OK then
begin t.InterfaceMethode; end; :( Fingolfin |
Alle Zeitangaben in WEZ +1. Es ist jetzt 10:27 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz