Delphi-PRAXiS
Seite 1 von 2  1 2      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   GUI-Design mit VCL / FireMonkey / Common Controls (https://www.delphipraxis.net/18-gui-design-mit-vcl-firemonkey-common-controls/)
-   -   Delphi FreeAndNil geht nicht (https://www.delphipraxis.net/39067-freeandnil-geht-nicht.html)

Hansa 28. Jan 2005 02:11


FreeAndNil geht nicht
 
Hi,

ich habe hier einen komplizierten Fall. :P Es geht um diese Stringgrid.Objects. Im Prinzip aber nur um TObject. Wenn ich diese Objects anlege, sie danach mit free freigebe, dann kommen seltsame Fehler. Und zwar daher, weil ich später nochmals auf NIL prüfe. Obwohl die Objekte mit free freigegeben wurden läuft das programm so, als wären sie zumindest nicht NIL. Ich setze sie deshalb zuerst auf NIL und dann erst rufe ich free auf und siehe an : es geht.

Kann mir das mal einer erklären ? :shock: Dieses Verhalten ist anscheinend auch die Ursache für die Existenz von FreeAndNil.

Das hier geht :

Delphi-Quellcode:
procedure TfrmEin.FormHide(Sender: TObject);
var i,j : integer;
begin
  for i := sg.FixedCols to sg.ColCount - 1 do  
    for j := sg.FixedRows to sg.RowCount - 1 do
      if sg.Objects [i,j] <> nil then begin
        sg.Objects [i,j] := nil;
        sg.Objects [i,j].Free;
      end;
end;
und hier :

Delphi-Quellcode:
procedure TfrmEin.FormHide(Sender: TObject);
var i,j : integer;
begin
  for i := sg.FixedCols to sg.ColCount - 1 do  
    for j := sg.FixedRows to sg.RowCount - 1 do
      if sg.Objects [i,j] <> nil then begin
        FreeAndNil (sg.Objects [i,j]);
      end;
end;
kommt folgender Fehler :

Zitat:

Zitat von D7-Compiler
[Fehler] ArtNrEin.pas(3118): Konstantenobjekt kann nicht als Var-Parameter weitergegeben werden


Luckie 28. Jan 2005 02:49

Re: FreeAndNil geht nicht
 
Die Methode Free entfernt nur das Objekt aus dem Speicher. Der Zeiger behält seine zugewiesene Adresse, die natürlich nach dem Aufruf von Free auf einen ungültigen Speicherbereich zeigt.

Hansa 28. Jan 2005 03:10

Re: FreeAndNil geht nicht
 
Das ist mir mittlerweile auch klar, aber wieso geht FreeAndNil nicht ? Die OH ist da äußerst sparsam. Hätte ich nicht von Hand die TObjects auf NIL gesetzt, dann wäre der Fehler immer noch da. :twisted:

Sprint 28. Jan 2005 03:28

Re: FreeAndNil geht nicht
 
Zitat:

Zitat von Hansa
Das ist mir mittlerweile auch klar, aber wieso geht FreeAndNil nicht?

Wie soll das denn gehen? FreeAndNil erwartet eine Variable in der eine Objektreferenz steht. Somit kann die Funktion die Variable ein "nichts" geben. Aber du willst keine Variable übergeben, sondern einen Wert der mit der Funktion GetObjects zurückgeben wird und intern aus einer Liste gelesen wird.

Robert Marquardt 28. Jan 2005 05:38

Re: FreeAndNil geht nicht
 
FreeAndNil hat einen var Parameter, daher kann man natuerlich keine Parameter uebergeben, die zu Funktionen evaluieren.
Eine Property mit Get- und Set-Methode kann daher nicht benutzt werden.

FreeAndNil hat noch einen Nebeneffekt. Erst wird der Parameter auf nil gesetzt und dann Free auf einer Kopie des
Parameters aufgerufen. Das kann zu AVs fuehren, wenn innerhalb des Destruktors aud die Parametervariable zugegriffen wird.

Also
MeinObjekt.Free;
MeinObjekt := nil;
benutzen wenn es mit FreeAndNil nicht geht.

teebee 28. Jan 2005 08:04

Re: FreeAndNil geht nicht
 
Zitat:

Zitat von Hansa
Das hier geht :

Delphi-Quellcode:
procedure TfrmEin.FormHide(Sender: TObject);
var i,j : integer;
begin
  for i := sg.FixedCols to sg.ColCount - 1 do  
    for j := sg.FixedRows to sg.RowCount - 1 do
      if sg.Objects [i,j] <> nil then begin
        sg.Objects [i,j] := nil;
        sg.Objects [i,j].Free;
      end;
end;

Das geht nur scheinbar, d.h. Du bekommst zwar keine Fehlermeldung aber Du erzeugst Speicherlecks. Free überprüft nämlich, ob die Referenz nil ist und ruft in diesem Fall Destroy _nicht_ auf. Die Überprüfung auf nil kannst Du Dir daher auch sparen.

Gruß, teebee

Robert Marquardt 28. Jan 2005 11:14

Re: FreeAndNil geht nicht
 
Das geht natuerlich nicht. Erst auf nil setzen und dann Free aufrufen ist falsch. :wall:

Hansa 28. Jan 2005 11:41

Re: FreeAndNil geht nicht
 
Und was mache ich jetzt ? Ich habe mir mal den Source von free angesehen :

Delphi-Quellcode:
procedure TObject.Free;
begin
  if Self <> nil then
    Destroy;
end;
Bei NIL passiert also tatsächlich rein gar nichts. Nun gut. Ich habe gedacht, dann nehme ich eben NIL und Destroy. Und weils so schön ist :lol: habe ich mir das dann auch noch in der RTL angeguckt :

Delphi-Quellcode:
destructor TObject.Destroy;
begin
end;
Und jetzt ? :shock: Fakt ist, daß das Programm so wie ich es gepostet habe so geht. Wie es aussieht weil ich eben den Wert auf NIL setze und ich ihn dadurch abfrage. In der Hilfe steht allerdings auch drin, man müsse selber diese Objects wieder freigeben.

Zitat:

Zitat von OH
Hinweis: Das TStringGridStrings-Objekt ist nicht der Eigentümer der Objekte des Arrays Objects. Objekte, die dem Array Objects hinzugefügt werden, sind auch dann noch vorhanden, wenn das TStringGridStrings-Objekt freigegeben wird. Sie müssen explizit von der Anwendung freigegeben werden.

:wiejetzt:

sakura 28. Jan 2005 11:43

Re: FreeAndNil geht nicht
 
So:
Delphi-Quellcode:
procedure TfrmEin.FormHide(Sender: TObject);
var i,j : integer;
begin
  for i := sg.FixedCols to sg.ColCount - 1 do  
    for j := sg.FixedRows to sg.RowCount - 1 do
      if sg.Objects [i,j] <> nil then begin
        sg.Objects [i,j].Free;
        sg.Objects [i,j] := nil;
      end;
end;
...:cat:...

Hansa 28. Jan 2005 12:06

Re: FreeAndNil geht nicht
 
So wurde mir das von anderer Stelle vorher auch schon geraten. :mrgreen: Mich stört hierbei aber das : "You may use this :" Ich bitte deshalb um Aufklärung, ob das jetzt so wasserdicht ist und warum. Insbesondere, weil ja das aufgerufebe Destroy leer ist. :wiejetzt: <-- der ist manchmal gut. :lol:


Alle Zeitangaben in WEZ +1. Es ist jetzt 10:41 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