AGB  ·  Datenschutz  ·  Impressum  







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

Meine neuste ARC-Hölle

Ein Thema von Mavarik · begonnen am 7. Feb 2018 · letzter Beitrag vom 7. Feb 2018
Antwort Antwort
Seite 1 von 2  1 2      
Benutzerbild von Mavarik
Mavarik

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

Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 16:42
Hallo Zusammen!

Wenn man ein Interface erzeugt und es wieder auf NIL setzt, was erwartet man dann? Richtig, dass der Destructor des Objects aufgerufen wird...

Auf Windows haben wir kein ARC, also was macht der gute App-Programmierer... Richtig keine Object verwenden, sondern unter allen Plattformen nur InterfacedObjects verwenden - das sollte ja überall gleich funktionieren...

Dachte ich...

Anbei ein kleines Testprogramm zum gruseln...

Mavarik
Angehängte Dateien
Dateityp: zip InterfaceProblem2.zip (2,7 KB, 8x aufgerufen)
  Mit Zitat antworten Zitat
Daniel
(Co-Admin)

Registriert seit: 30. Mai 2002
Ort: Hamburg
13.920 Beiträge
 
Delphi 10.4 Sydney
 
#2

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17:20
Mit einem Beitrag wie diesem erzeugst Du eigentlich nur Unsicherheit und Halbwissen auf allen Seiten. Hast Du analysiert, an welcher Stelle das von Dir beobachtete Verhalten vom erwarteten Verhalten abweicht? Oder ist es genau das, was Du im Rahmen dieser Diskussion herausfinden möchtest?
Daniel R. Wolf
mit Grüßen aus Hamburg
  Mit Zitat antworten Zitat
Benutzerbild von Uwe Raabe
Uwe Raabe

Registriert seit: 20. Jan 2006
Ort: Lübbecke
11.475 Beiträge
 
Delphi 12 Athens
 
#3

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17:43
Zitat:
Wenn man ein Interface erzeugt und es wieder auf NIL setzt, was erwartet man dann? Richtig, dass der Destructor des Objects aufgerufen wird...
Das ist vielleicht etwas zu salopp formuliert. Bei Reference-Counted Instanzen wird der Destructor aufgerufen, wenn der RefCount auf 0 geht. Das kann mit dem expliziten Nil-Setzen einer Interface-Variable einher gehen, muss es aber nicht.

In dem Fall Button1 vermute ich, du spielst auf die verspätete Freigabe an, die erst beim Beenden der Routine erfolgt. Hier kommen drei Dinge zusammen, die zu diesem Verhalten führen:
  1. GFoo ist eine globale Variable
  2. Das Interface wird durch eine Funktion erzeugt
  3. Beide Zuweisungen auf GFoo findet im selben Scope statt

In Punkt 2 erzeugt der Compiler in dieser Konstellation eine versteckte Variable für den Rückgabewert und weist diesen getrennt der globalen Variablen zu. Diese versteckte Variable wird aber eben erst bei Verlassen des Scope auf nil gesetzt.

Was Button2 betrifft, kann ich keine Besonderheit erkennen. Was erwartest du denn dort und was passiert bei dir?
Uwe Raabe
Certified Delphi Master Developer
Embarcadero MVP
Blog: The Art of Delphi Programming
  Mit Zitat antworten Zitat
Benutzerbild von Stevie
Stevie

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

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17:51
Zirkuläre Referenz - der Aufruf von FInvoke in der anonymen Methode in TFDKMessageHandler captured Self implizit und sorgt unter ARC somit für ein __ObjAddRef -> memory leak

Ist schon länger bekannt, dass anonyme Methoden unter ARC schnell mal Leaks erzeugen - https://quality.embarcadero.com/browse/RSP-10176

Was wir in Spring4D typischerweise an solchen Stellen machen, ist folgendes:

Delphi-Quellcode:
constructor TFDKMessageHandler<TMSG>.Create(AInvoke: TMyMSGType);
{$IFDEF AUTOREFCOUNT}
var
  capturedSelf: Pointer; // alternativ mit [unsafe] dann spart man sich den hardcast weiter unten, aber da gabs in früheren Versionen noch Probleme, daher stumpf mit Pointer
{$ENDIF}
begin
  inherited Create;
  FInvoke := AInvoke;

{$IFDEF AUTOREFCOUNT}
  capturedSelf := Self;
{$ENDIF}

  FID := TMessageManager.DefaultManager.SubscribeToMessage(TFDKMSGType<TMSG>,Procedure (Const Sender : TObject;Const M : TMessage)
           begin
           {$IFDEF AUTOREFCOUNT}with TFDKMessageHandler<TMSG>(capturedSelf) do{$ENDIF}
             FInvoke(Sender,TMSG(TFDKMSGType<TMSG>(M).Value));
           end);
end;
Stefan
“Simplicity, carried to the extreme, becomes elegance.” Jon Franklin

Delphi Sorcery - DSharp - Spring4D - TestInsight

Geändert von Stevie ( 7. Feb 2018 um 17:59 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Uwe Raabe
Uwe Raabe

Registriert seit: 20. Jan 2006
Ort: Lübbecke
11.475 Beiträge
 
Delphi 12 Athens
 
#5

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17:56
Zirkuläre Referenz
Wäre ja auch wirklich kein Luxus gewesen, das erwartete und tatsächliche Verhalten mal eben zu beschreiben.
Uwe Raabe
Certified Delphi Master Developer
Embarcadero MVP
Blog: The Art of Delphi Programming
  Mit Zitat antworten Zitat
Benutzerbild von Stevie
Stevie

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

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 18:04
Zirkuläre Referenz
Wäre ja auch wirklich kein Luxus gewesen, das erwartete und tatsächliche Verhalten mal eben zu beschreiben.
Ich dachte, das wüsstest du von der anderen Stelle, wo Frank das heute schon gepostet hat
Stefan
“Simplicity, carried to the extreme, becomes elegance.” Jon Franklin

Delphi Sorcery - DSharp - Spring4D - TestInsight
  Mit Zitat antworten Zitat
Benutzerbild von Uwe Raabe
Uwe Raabe

Registriert seit: 20. Jan 2006
Ort: Lübbecke
11.475 Beiträge
 
Delphi 12 Athens
 
#7

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 18:09
Ich dachte, das wüsstest du von der anderen Stelle, wo Frank das heute schon gepostet hat
Da kann aber nun mal nicht jeder reinschauen und wir beide (und vielleicht noch eine Handvoll andere) sind hier wohl auch nicht die Zielgruppe gewesen, oder?
Uwe Raabe
Certified Delphi Master Developer
Embarcadero MVP
Blog: The Art of Delphi Programming
  Mit Zitat antworten Zitat
Benutzerbild von Mavarik
Mavarik

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

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 18:20
Mit einem Beitrag wie diesem erzeugst Du eigentlich nur Unsicherheit und Halbwissen auf allen Seiten. Hast Du analysiert, an welcher Stelle das von Dir beobachtete Verhalten vom erwarteten Verhalten abweicht? Oder ist es genau das, was Du im Rahmen dieser Diskussion herausfinden möchtest?
Nun ja, sagen wir mal so... Ich versuche seit zwei Wochen die "letzten" Fehler aus meiner App zu bekommen, stoße aber leider immer wieder auf Probleme, die sich nicht eindeutig Identifizieren lassen.

Hierbei gehe ich immer erstmal davon aus:
1. Der Fehler liegt bei mir
2. Der Fehler liegt an der aktuellen Delphi-Version
3. Es ist ein genereller Fehler

Und dann stellt sich immer die Frage QC oder nicht...

Daher tausche ich mich "gerne" vorher mit dem ein oder anderen aus.


Wäre ja auch wirklich kein Luxus gewesen, das erwartete und tatsächliche Verhalten mal eben zu beschreiben.
Da gebe ich Dir recht... Wollte eigentlich mehr schreiben, aber dann ging das Telefon - Du kennst das sicherlich auch...

Ist schon länger bekannt, dass anonyme Methoden unter ARC schnell mal Leaks erzeugen - https://quality.embarcadero.com/browse/RSP-10176
Danke für Deine Hilfe (überall )

Mein kleines Testprogramm - abgesehen von den Erklärungen von Uwe und Stefan, die natürlich richtig sind - zeigt aber sicherlich auf, dass sich das Verhalten unter ARC nicht immer so offensichtlich zeigt, wir man es erwartet.

Mavarik
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

Registriert seit: 11. Okt 2003
Ort: Elbflorenz
44.184 Beiträge
 
Delphi 12 Athens
 
#9

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 18:38
Und jetzt versuch mal so ein DOM hinzubekommen, dass unter ARC und NichtARC funktioniert und wo es massig Kreisreferenzen gibt, ohne ein Speicherleck.

Parent kennt Kinder und Kinder kennen Parent und Root.



Und das ohne böse Casts, um die Referenzählung des ARC zu umgehen
und so, dass der Code deiner Componente auch mit etwas älteren Delphis noch kompatibel ist,
und ohne dass der halbe Code nur noch aus IFDEFS besteht.

[WeakRef] kann/konnte man nicht verwenden, da es das unter Windows nicht gibt/gab.
Und [NoRef] vermisse ich sowieso.
$2B or not $2B

Geändert von himitsu ( 7. Feb 2018 um 18:42 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Stevie
Stevie

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

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 18:42
Hier nochmal ein Minimalbeispiel (ich mach beim ARC Zeugs testen seit 10.2 immer gern Programme die ich kurz auf meine Ubuntu VM feuern kann, daher ohne FMX)
Delphi-Quellcode:
program Project1;

{$APPTYPE CONSOLE}

uses
  System.SysUtils;

type
  TTest = class(TInterfacedObject)
  private
    fProc: TProc;
    procedure Something;
  public
    constructor Create;
    destructor Destroy; override;
  end;

var
  destroyCalled: Boolean;

{ TTest }

constructor TTest.Create;
begin
  fProc := Something;
(*
  actually compiled into

  fProc :=
    procedure begin
    begin
      Self.Something; // <- causes Self to be captured
      // all captured fields are strong references and thus TTest keeps itself alife
      // with the circular reference: fProc <-> Self
    end;
    *)

end;

destructor TTest.Destroy;
begin
  destroyCalled := True;
  inherited;
end;

procedure TTest.Something;
begin
end;

procedure Main;
var
  intf: IInterface;
begin
  intf := TTest.Create;
  intf := nil;
  Assert(destroyCalled);
end;

begin
  Main;
end.
Da anonyme Methoden immer mehr Einzug halten (auch im offiziellen Delphi Code) halte ich diesen Defekt schon für äußerst
kritisch (ja, man kann drumherum arbeiten) gerade da er schon einiges an Wissen über die Interna vorraussetzt.

Also bitte fleißig voten sofern möglich

Und jetzt versuch mal so ein DOM hinzubekommen, dass unter ARC und NichtARC funktioniert und wo es massig Kreisreferenzen gibt, ohne ein Speicherleck.
Siehe FMX und FireDAC :grusel:
Stefan
“Simplicity, carried to the extreme, becomes elegance.” Jon Franklin

Delphi Sorcery - DSharp - Spring4D - TestInsight

Geändert von Stevie ( 7. Feb 2018 um 18:44 Uhr)
  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 11:46 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