Delphi-PRAXiS
Seite 1 von 2  1 2      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Refcount meines Objektes geht zu früh auf 0 (https://www.delphipraxis.net/198318-refcount-meines-objektes-geht-zu-frueh-auf-0-a.html)

ngott2 24. Okt 2018 08:04

Delphi-Version: 10 Seattle

Refcount meines Objektes geht zu früh auf 0
 
Hallo,
ich habe eine Funktion, wo ich ein Interface übergebe. Soweit so gut. Bevor ich es als Prameter übergebe habe ich eine Liste von Objekten die von meinem Interface und TInterfaceobject erben. Nun rufe ich meine Funktion auf die das Interface erwartet mit folgenden befehl
Delphi-Quellcode:
 if meineFunktion(meineListe[lindex] as myInterface) then
.
Die Funktion läuft ohne Probleme durch. Am Ende der Funktion wird der Refcount, aber auf 0 gesetzt und mein Objekt wird freigeben. Warum wird mein Refcount auf 0 gesetzt, da mein Objekt nocht in meiner TObjectlist hinterlegt ist sollte der Refcount auf 1 bleiben. Wenn ich jetzt meine Liste mit Owned Objects freigebe fliegt es mir um die Ohren weil er versucht die Objekte freizugeben. Hat jemand eine Idee?


Gruß,
ngott2

TiGü 24. Okt 2018 08:30

AW: Refcount meines Objektes geht zu früh auf 0
 
Zitat:

Zitat von ngott2 (Beitrag 1416464)
. Warum wird mein Refcount auf 0 gesetzt, da mein Objekt nocht in meiner TObjectlist hinterlegt ist sollte der Refcount auf 1 bleiben. Wenn ich jetzt meine Liste mit Owned Objects freigebe fliegt es mir um die Ohren weil er versucht die Objekte freizugeben. Hat jemand eine Idee?

"Geinterfacete" Objekte nur als Interfaces behandeln und nicht als Objekt anfassen (außer man weiß was man tut).
Erster Schritt: Aus der Objektlist entfernen. Wenn Listeneintrag gebraucht, dann mach dir eine TList<IDeinInterface>, in denen du die Interface-Referenz hältst.

mjustin 24. Okt 2018 08:31

AW: Refcount meines Objektes geht zu früh auf 0
 
Versuche dies:

Delphi-Quellcode:
var
 Tmp: myInterface;
begin
 ...
 Tmp := meineListe[lindex] as myInterface;
 if meineFunktion(Tmp) then
 ...

dummzeuch 24. Okt 2018 11:10

AW: Refcount meines Objektes geht zu früh auf 0
 
Zitat:

Zitat von mjustin (Beitrag 1416475)
Versuche dies:

Delphi-Quellcode:
var
 Tmp: myInterface;
begin
 ...
 Tmp := meineListe[lindex] as myInterface;
 if meineFunktion(Tmp) then
 ...

Wird nicht funktionieren, dann geht der Reference-Counter zwar nicht auf 0, wenn die Funktion endet, aber dafür dann, wenn tmp nicht mehr gebraucht wird.

dummzeuch 24. Okt 2018 11:24

AW: Refcount meines Objektes geht zu früh auf 0
 
Es gibt für das Problem zwei mögliche Lösungen:
  1. Nicht das Objekt sondern ein Interface in der Liste speichern.
  2. Die Implementation von _AddRef und _Release des Objekts ändern, so dass es kein Reference-Counting mehr durchführt (das macht z.B. TComponent so (oder war es bereits TPersistent?))
  3. Absolut nicht zu emfpehlen: Den Reference-Counter direkt manipulieren. Nach der Erzeugung auf 1 setzen, vor dem Freigeben auf 0 setzen. (Ja, das ist ein Hack und wie gesagt nicht zu empfehlen.)

hoika 24. Okt 2018 12:13

AW: Refcount meines Objektes geht zu früh auf 0
 
Hallo,
die Variable wurde auch als Const übergeben?

ngott2 24. Okt 2018 13:41

AW: Refcount meines Objektes geht zu früh auf 0
 
Zitat:

Zitat von hoika (Beitrag 1416524)
Hallo,
die Variable wurde auch als Const übergeben?

Ja übergebe ich als const.


Ich werde mal alles umbauen. Normalerweise arbeite ich auch immer mit den Interface, aber jemand anders hat die Basis geschrieben und leider nur mit Klassen gearbeitet.

Würde aber trotzdem gerne verstehen, wie Delphi das Intern macht. Weil wenn ich es mit ein Interface mache dann fügt er ja erst eine Referenz hinzu und löscht sie am Ende der Methode wieder. Warum funktioniert dies nicht gemischt Zwischen Interface und Klasse?

Bernhard Geyer 24. Okt 2018 14:06

AW: Refcount meines Objektes geht zu früh auf 0
 
Zitat:

Zitat von ngott2 (Beitrag 1416540)
Würde aber trotzdem gerne verstehen, wie Delphi das Intern macht. Weil wenn ich es mit ein Interface mache dann fügt er ja erst eine Referenz hinzu und löscht sie am Ende der Methode wieder. Warum funktioniert dies nicht gemischt Zwischen Interface und Klasse?

Wenn es kein Interface mehr auf die Instanz gibt wird gnadenlos abgeräumt, egal ob es noch einen Zeiger auf die Klasse gibt.

Ein "Mischbetrieb" sollte man nur im absoluten Notfall verwenden. Und dann funktioniert es nur wenn man explizit einmal _AddRef aufruft um die automatische Referenzzählung auszuhebeln.

Uwe Raabe 24. Okt 2018 14:08

AW: Refcount meines Objektes geht zu früh auf 0
 
Wenn man die Referenzzählung nicht will, darf man nicht von
Delphi-Quellcode:
TInterfacedObject
ableiten, sondern von
Delphi-Quellcode:
TInterfacedPeristent
.

stahli 24. Okt 2018 14:48

AW: Refcount meines Objektes geht zu früh auf 0
 
@ngott2

Du darfst auch nicht folgendes machen:

Delphi-Quellcode:
procedure DoIt(Intf: IInterface);
begin
end;

...

DoIt(TMyClass.Create); // falsch!

...

var
tmpIntf: IInterface;
 
tmpIntf := TMyClass.Create;
DoIt(tmpIntf); // korrekt

In der ersten Variante kommt auch die Referenzzählung durcheinander.
Das ist etwas unschön und man muss das erst mal wissen.

Aber wenn man das beachtet, kann man damit schon vernünftig leben.

Ich arbeite inzwischen nur noch mit Interfaces und verstecke die Klassen hinter einer Factory. Dann kann man solche Probleme grundsätzlich ausschließen.


Alle Zeitangaben in WEZ +1. Es ist jetzt 20:10 Uhr.
Seite 1 von 2  1 2      

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