AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Thema durchsuchen
Ansicht
Themen-Optionen

Freigabe von übergebenen Interface-Instanzen

Ein Thema von AndyDF · begonnen am 30. Sep 2016 · letzter Beitrag vom 30. Sep 2016
Antwort Antwort
Seite 1 von 2  1 2      
AndyDF

Registriert seit: 6. Sep 2006
Ort: Allgäu
99 Beiträge
 
Delphi 10.4 Sydney
 
#1

Freigabe von übergebenen Interface-Instanzen

  Alt 30. Sep 2016, 10:12
Delphi-Version: 10 Seattle
Habe gerade etwas fest gestellt, von dem ich bisher erwartet hatte, dass es funktioniert. Ich arbeite sehr intensiv mit Interfaces und schaue auch immer, dass ich Objekt- und Interfacereferenzen nicht vermische.
Das heißt, Variable vom Interface-Typ definierten und das Objekt mittels Create der Variable zuweisen. Wenn die Interface-Variable aus dem Scope fällt, wird das Objekt dahinter auch immer schön frei gegeben. So soll das sein!

Jetzt hätte ich erwartet, wenn eine Methode eine Interface-Variable übernimmt, und ich beim Aufruf direkt die neu erstelle Objekt-Referenz übergebe, dass der RefCount und die Freigabe korrekt funktionieren. Dem ist anscheinend nicht so. Im Folgenden mal ein Beispiel-Code. Habe an der entsprechenden Stelle einen Kommentar eingefügt...


Delphi-Quellcode:
type
  IMyInterface = interface(IInterface)
    ['{475ACE39-ECC7-467D-A325-35B2DF22203A}']
    procedure DoSomething;
  end;

  TMyInterface = class(TInterfacedObject, IMyInterface)
  private
    procedure DoSomething;
  public
    destructor Destroy; override;
  end;

  TForm1 = class(TForm)
    Button1: TButton;
    procedure Button1Click(Sender: TObject);
  private
    procedure Machwas(const AValue: IMyInterface);
  public
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

destructor TMyInterface.Destroy;
begin
  ShowMessage('Free!');
end;

procedure TMyInterface.DoSomething;
begin
  //...
end;

procedure TForm1.Button1Click(Sender: TObject);
var
  Value: IMyInterface;
begin
  //Wenn der Aufruf auf diese Weise erfolgt, wird das Objekt NICHT frei gegeben.
  Machwas(TMyInterface.Create);

  //So wird das Objekt korrekt frei gegeben
  Machwas(TMyInterface.Create as IMyInterface);

  //so wird das Objekt auch korrekt frei geben
  Value := TMyInterface.Create;
  Machwas(Value);
end;

procedure TForm1.Machwas(const AValue: IMyInterface);
begin
  AValue.DoSomething;
end;
Ist das logisch? Hätte ich bisher anders erwartet... da mir aber der Speicher voll lief, wurde ich eines besseren belehrt.
Andreas Blenk
  Mit Zitat antworten Zitat
Der schöne Günther

Registriert seit: 6. Mär 2013
6.156 Beiträge
 
Delphi 10 Seattle Enterprise
 
#2

AW: Freigabe von übergebenen Interface-Instanzen

  Alt 30. Sep 2016, 10:16
Das ist ein abartiger Fallstrick. Das const in deiner Methode bedeutet für den Compiler "Die Referenz wird nicht verändert. Ach komm, sparen wir uns gleich die Referenzzählung". Tödlich, wie du siehst.

Ich finde es auch schrecklich, die einzige Hilfe ist sich das eigentlich sinnvolle const bei Interface-Argumenten abzugewöhnen.
  Mit Zitat antworten Zitat
AndyDF

Registriert seit: 6. Sep 2006
Ort: Allgäu
99 Beiträge
 
Delphi 10.4 Sydney
 
#3

AW: Freigabe von übergebenen Interface-Instanzen

  Alt 30. Sep 2016, 10:19
Das ist ein abartiger Fallstrick. Das const in deiner Methode bedeutet für den Compiler "Die Referenz wird nicht verändert. Ach komm, sparen wir uns gleich die Referenzzählung". Tödlich, wie du siehst.

Ich finde es auch schrecklich, die einzige Hilfe ist sich das eigentlich sinnvolle const bei Interface-Argumenten abzugewöhnen.
Hmm... wenn du mir das so schreibst, ist eigentlich logisch. Bin ich klassisch drauf rein gefallen.
Danke für die Info.
Andreas Blenk
  Mit Zitat antworten Zitat
mjustin

Registriert seit: 14. Apr 2008
3.006 Beiträge
 
Delphi 2009 Professional
 
#4

AW: Freigabe von übergebenen Interface-Instanzen

  Alt 30. Sep 2016, 10:54
Ich habe mir das Machwas(TMyInterface.Create); abgewöhnt.

Seitdem funktioniert's

Das const lasse ich daher. Es soll m.W. ein paar CPU Anweisungen einsparen.
Michael Justin
  Mit Zitat antworten Zitat
Benutzerbild von jaenicke
jaenicke

Registriert seit: 10. Jun 2003
Ort: Berlin
9.580 Beiträge
 
Delphi 11 Alexandria
 
#5

AW: Freigabe von übergebenen Interface-Instanzen

  Alt 30. Sep 2016, 11:25
Das const ist bei häufigen Aufrufen spürbar schneller, ja.

Das direkte Übergeben macht man eben nicht. Wenn man es so schreibt, geht es ja. Das hast du ja jetzt vermutlich auch so:
Delphi-Quellcode:
var
  Value: IMyInterface;
begin
  Value := TMyInterface.Create;
  Machwas(Value);
Sebastian Jänicke
Alle eigenen Projekte sind eingestellt, ebenso meine Homepage, Downloadlinks usw. im Forum bleiben aktiv!
  Mit Zitat antworten Zitat
Fritzew

Registriert seit: 18. Nov 2015
Ort: Kehl
678 Beiträge
 
Delphi 11 Alexandria
 
#6

AW: Freigabe von übergebenen Interface-Instanzen

  Alt 30. Sep 2016, 11:26
Ja das ist eine "unschönheit" im Delphi Compiler. Ist nicht konsistent in der Hinsicht. Wurde bei stackoverflow auch schon
hoch und runter diskutiert. Zumindest ein Hint sollte da auftauchen.
http://stackoverflow.com/questions/4...ly-as-const-in
Gruss Fritz
PS: Hallo Andreas
Fritz Westermann
  Mit Zitat antworten Zitat
AndyDF

Registriert seit: 6. Sep 2006
Ort: Allgäu
99 Beiträge
 
Delphi 10.4 Sydney
 
#7

AW: Freigabe von übergebenen Interface-Instanzen

  Alt 30. Sep 2016, 11:36
Immerhin geht ja folgendes:

Machwas(TMyInterface.Create as IMyInterface);

Dann muss man nicht unbedingt eine Variable anlegen. Allerdings muss man dran denken....

Gruß Andreas
PS: Hallo Fritz.
Andreas Blenk
  Mit Zitat antworten Zitat
Benutzerbild von Stevie
Stevie

Registriert seit: 12. Aug 2003
Ort: Soest
4.016 Beiträge
 
Delphi 10.1 Berlin Enterprise
 
#8

AW: Freigabe von übergebenen Interface-Instanzen

  Alt 30. Sep 2016, 11:41
In diese Falle sind schon Generationen von Delphi Entwicklern getappt:

http://qc.embarcadero.com/wc/qcmain.aspx?d=90482
https://quality.embarcadero.com/browse/RSP-10100
https://quality.embarcadero.com/browse/RSP-15797

Aber trotzdem wird es von den Verantwortlichen leider immernoch wegdiskutiert:

https://plus.google.com/+StefanGlien...ts/SM1oVCNa98q
Stefan
“Simplicity, carried to the extreme, becomes elegance.” Jon Franklin

Delphi Sorcery - DSharp - Spring4D - TestInsight

Geändert von Stevie (30. Sep 2016 um 11:47 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Mavarik
Mavarik

Registriert seit: 9. Feb 2006
Ort: Stolberg (Rhld)
4.142 Beiträge
 
Delphi 10.3 Rio
 
#9

AW: Freigabe von übergebenen Interface-Instanzen

  Alt 30. Sep 2016, 13:00
Wenn ich keine Factory für die Klasse habe, mache ich es einfach so...

Delphi-Quellcode:
type
  TMyInterface = class(TInterfacedObject, IMyInterface)
  private
    procedure DoSomething;
  public
    destructor Destroy; override;
    Class Function Construct : IMyInterface;
  end;

implementation


procedure TForm1.Button1Click(Sender: TObject);
begin
  Machwas(TMyInterface.Construct);
end;
Und läuft...
  Mit Zitat antworten Zitat
Fritzew

Registriert seit: 18. Nov 2015
Ort: Kehl
678 Beiträge
 
Delphi 11 Alexandria
 
#10

AW: Freigabe von übergebenen Interface-Instanzen

  Alt 30. Sep 2016, 13:11
Was machst Du dann bei Parametern im Constructor? Alles doppelt pflegen?
Und wie gewährleistest Du das die Benutzer Deiner Klasse das auch so anwenden?
Fritz Westermann
  Mit Zitat antworten Zitat
Antwort Antwort
Seite 1 von 2  1 2      


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 09:29 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