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 Stevie
Stevie

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

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 16: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 16:59 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Uwe Raabe
Uwe Raabe

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

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 16: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.052 Beiträge
 
Delphi 10.1 Berlin Enterprise
 
#3

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17: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.785 Beiträge
 
Delphi 12 Athens
 
#4

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17: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.165 Beiträge
 
Delphi 10.3 Rio
 
#5

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?
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.588 Beiträge
 
Delphi 12 Athens
 
#6

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17: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.
Ein Therapeut entspricht 1024 Gigapeut.

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

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

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17: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 17:44 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

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

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17:45
Zitat:
intf := nil;
Wenn du statt an ARC (Automatic Reference Counting) an das in meiner Signatur denkst (Garbage Collector),
dann hieße es "ich brauch dich nicht mehr ... kannst du böser SpeicherManager also irgndwann wegwerfen".
Ein Therapeut entspricht 1024 Gigapeut.

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

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

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17:49
Wenn du statt an ARC (Automatic Reference Counting) an das in meiner Signatur denkst (Garbage Collector),
dann hieße es "ich brauch dich nicht mehr ... kannst du also also irgndwann wegwerfen".
Nein, es heißt: "Diese Variable zeigt nicht mehr auf dich, wenn sonst auch keiner mehr auf dich zeigt, dann ..."
Stefan
“Simplicity, carried to the extreme, becomes elegance.” Jon Franklin

Delphi Sorcery - DSharp - Spring4D - TestInsight
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

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

AW: Meine neuste ARC-Hölle

  Alt 7. Feb 2018, 17:53
"... dann braucht dich keiner mehr"

Aber das heißt noch lange nicht, dass man es auch sofort das Freigeben muß. (aktuell im ARC ja, aber wer weiß was kommt)
Es könnte auch erstmal nur als Frei markiert werden und später kümmert sich in Ruhe jemand ums Aufräumen .... so ist das Programm gleich viel schneller und braucht nur ein bissl mehr Speicher.

Wenn du den alten Kühlschrank nicht mehr brauchst, dann trägst du ihn doch auch nicht sofort zum Wertstoffhof?
Ein Therapeut entspricht 1024 Gigapeut.

Geändert von himitsu ( 7. Feb 2018 um 18:01 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 07:24 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-2025 by Thomas Breitkreuz