Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Delphi Interfaces: Ich werd' irre! (https://www.delphipraxis.net/23738-interfaces-ich-werd-irre.html)

Fingolfin 8. Jun 2004 20:28


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:
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;
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.

: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

Bernhard Geyer 8. Jun 2004 21:04

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:
var
  FTest: ITest;
Lösung 2: Zurücksetzen der Modulvariable im Destruktur
Delphi-Quellcode:
destructor TTest.Destroy;
begin
  FTest := nil;
  inherited;
end;
Lösung nicht getestet, jedoch aufgrund eigener Erfahrungen müsste das dein Problem lösen.

Fingolfin 8. Jun 2004 21:11

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

Fingolfin 10. Jul 2004 19:07

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

Fingolfin 11. Jul 2004 14:01

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:
  ITest = interface
  ['{93942268-B484-4BBD-A907-CA8B8B5208BF}']
  end;

  TTest = class(TInterfacedObject, ITest)
  end;
Dann erstelle ich mein Objekt:

Code:
var
  test: TTest;
begin
  test := TTest.Create;
end;
Und jetzt möchte ich per GetInterface wissen, ob mein Objekt das ITest interface implementiert:

Code:
var
  testint: ITest;
begin
 if test.GetInterface(itest, testint) then
  begin
    //hier geht schon nix mehr, weil test freigegeben wurde
  end;
end;
:wall:

Das ist doch ein Grundlagen-Problem oder? Irgendwie fehlt mir der Durchblick.

Hilfe :cry:

Danke im Voraus,

Fingolfin

Bernhard Geyer 11. Jul 2004 17:07

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

Fingolfin 11. Jul 2004 19:22

Re: Interfaces: Ich werd' irre!
 
Danke für deinen Kommentar.

Code:
(test as ITest).InterfaceMethode
funktioniert ohne Probleme.

Die Abfrage:

Code:
if test is ITest then
führt aber leider zu einem "Operator not applicable to this operand type" Fehler. :(

@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

mirage228 11. Jul 2004 19:25

Re: Interfaces: Ich werd' irre!
 
Zitat:

Zitat von Fingolfin
Die Abfrage:

Code:
if test is ITest then
führt aber leider zu einem "Operator not applicable to this operand type" Fehler. :(


Fingolfin

Hi,

dazu könntest du dir die Methode Hier im Forum suchenQueryInterface in der OH anschauen.

mfG
mirage228

Fingolfin 11. Jul 2004 19:45

Re: Interfaces: Ich werd' irre!
 
Hallo mirage. Mit

Code:
  if QueryInterface(ITest, t) = S_OK then
  begin
    t.InterfaceMethode;
  end;
funktioniert es.

Allerdings steht in der OH ja, daß man gerade QueryInterface vermeiden und GetInterface nutzen soll. GetInterface macht eigentlich auch nur Folgendes:

Code:
  if GetInterface(IID, Obj) then
    Result := S_OK
  else
    Result := E_NOINTERFACE;
Was an der QueryInterface-Variante dann auch noch komisch ist, ist die Tatsache, daß ein danach ausgeführtes

Code:
  t := nil;
anscheinden nix bewirkt.

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

Fingolfin 11. Jul 2004 19:58

Re: Interfaces: Ich werd' irre!
 
Oha, habe natürlich Unsinn erzählt.

Anstatt:

Code:
  if QueryInterface(ITest, t) = S_OK then
  begin
    t.InterfaceMethode;
  end;
muß es natürlich:

Code:
  if test.QueryInterface(ITest, t) = S_OK then
  begin
    t.InterfaceMethode;
  end;
heißen. :wall: Und das funktioniert genausowenig wie QueryInterface.

:(

Fingolfin


Alle Zeitangaben in WEZ +1. Es ist jetzt 05:15 Uhr.

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz