Delphi-PRAXiS
Seite 2 von 4     12 34      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Rekursives Freigeben einer TStringList // Compilerfehler (https://www.delphipraxis.net/204647-rekursives-freigeben-einer-tstringlist-compilerfehler.html)

Caps 15. Jun 2020 13:24

AW: Rekursives Freigeben einer TStringList // Compilerfehler
 
Leider greife ich hinterher noch drauf zu, vorher prüfe ich aber, ob die Liste nil ist, und da hat's geknallt ^^.

Lemmy 15. Jun 2020 15:16

AW: Rekursives Freigeben einer TStringList // Compilerfehler
 
um es mal zu umschreiben:
es ist keine gute Idee, erst die Wohnung in die Luft zu jagen und sich dann zu wundern, dass man mit dem Festnetzanschluss bei der Feuerwehr nicht mehr durch kommt.

Wenn Du also die TStringList incl. den Kind-Elementen frei gibts - welchen genauen Grund mag es geben anschließend nochmal auf die Kindelemente zuzugreifen? An der Stelle liegt dein Fehler, nicht aber darin ob bei der Freigabe der Kindelemente nun NIL zugewiesen wurde oder eben nicht...

Caps 15. Jun 2020 16:03

AW: Rekursives Freigeben einer TStringList // Compilerfehler
 
Ja, ich weiß was du meinst.
Die Stelle war ebenfalls ein Destruktor, der nochmal geprüft hat, ob das Objekt nil ist, und wenn nicht, Free aufruft. Du hast schon recht, dass das eigentlich nicht nötig sein sollte, es ist dennoch momentan so.
Mein Problem war das rekursive Freigeben einer Stringliste.

Grüße!

himitsu 15. Jun 2020 16:09

AW: Rekursives Freigeben einer TStringList // Compilerfehler
 
Wenn man nur mit TComponents arbeitet, dann kann man deren Notifications benutzen.

Da können sich Andere bei der Klasse registrieren und werden informiert, wenn die Komponente freigegeben wird, um die Referenz bei sich zu entfernen.
z.B. TEdit<->TDataSource<->TDataSet oder TButton<->TImageList/TPopup, ....
Man löscht die ImageList und das Property in der anderen Komponente wird NIL.

Es gibt auch eine TComponentList, die das nutzt.
Da kann man bei Verschwinden der Komponente den Eintrag in der Liste auf NIL setzen oder gleich komplett entfernen lassen.

Delphi.Narium 15. Jun 2020 16:33

AW: Rekursives Freigeben einer TStringList // Compilerfehler
 
Ursprünglich ging es doch eigentlich darum, dass es einen Kompilierfehler gab. Bei meinem ollen Delphi 7 gibt es den mit diesem veränderten Code nicht mehr. Reicht das zur Problemlösung?
Delphi-Quellcode:
procedure FreeStringList(var List: TStringList);
var
   i     : Integer;
   myList : TStringList;
begin
  if Assigned(List) then begin
    for i := 0 to List.Count - 1 do begin
      if Assigned(List.Objects[i]) then begin
        if List.Objects[i] is TStringList then begin
          myList := TStringList(List.Objects[i]);
          // Unterknoten freigeben ...
          FreeStringList(myList);
        end else begin
          // Blatt freigeben ...
          FreeAndNil(List.Objects[i]);
        end;
      end;
    end;
    FreeAndNil(List);
  end;
end;
Wenn ein Funktionsparameter mit var angegeben wird, dann kann man da halt keinen Cast ala TStringList(irgendeinobjekt) reingeben (derweil das ist dann const), man muss sich die Mühe machen, dort eine Variabel zu übergeben. Es ist also der Aufwand einer Variabelndeklaration des passenden Types und eine Zuweisung des Casts TStringList(irgendeinobjekt) an die entsprechende Variabel nötig. Die Variabel kann man dann problemlos in der Funktionsaufruf schreiben und der Compiler ist's zufrieden.

Und wie immer: Man kann grundsätzlich jedes Problem auch irgendwie anders lösen, aber vielleicht reicht diese einfache Änderung ja auch aus ;-)

Caps 16. Jun 2020 07:52

AW: Rekursives Freigeben einer TStringList // Compilerfehler
 
Moin,

nur zur Unterhaltung: es geht doch, wenn auch nur mit abenteuerlichem Herumgecaste:

Delphi-Quellcode:
procedure FreeStringList(var List);
var
  i: Integer;
  Obj, Obj_: TObject;
  List_: TStringList;
begin
  Obj := TObject(List);
  if Obj <> nil then begin
    if Obj is TStringList then begin
      List_ := TStringList(Obj);
      with List_ do begin
        if Count > 0 then begin
          for i:=0 to Count-1 do begin
            if Objects [i] <> nil then begin
              Obj_ := Objects [i];
              FreeStringList(Obj_);
            end;
          end;
        end;
      end;
    end;
    TObject(List).Free;
    TObject(List) := nil;
  end;
end;
lg Caps

Uwe Raabe 16. Jun 2020 08:54

AW: Rekursives Freigeben einer TStringList // Compilerfehler
 
Dieser Code
Delphi-Quellcode:
              Obj_ := Objects [i];
              FreeStringList(Obj_);
sorgt nicht dafür, das
Delphi-Quellcode:
Objects[i]
hinterher nil ist! Lediglich
Delphi-Quellcode:
Obj_
ist dann nil.

Die
Delphi-Quellcode:
Objects[]
Werte in einer TStringList lassen sich nur durch eine direkte Zuweisung auf nil setzen. Eine Übergabe als var Parameter, ob typisiert oder untypisiert, lässt der Compiler aus genau diesem Grund nicht zu.

Caps 16. Jun 2020 10:04

AW: Rekursives Freigeben einer TStringList // Compilerfehler
 
Stimmt, ich merk's. Schade ^^

Caps 16. Jun 2020 11:00

AW: Rekursives Freigeben einer TStringList // Compilerfehler
 
Der Vollständigkeit halber meine jetzige Lösung, falls jemand ein ähnliches Problem hat:

Neuer Typ:
Delphi-Quellcode:
TStringTree = class(TStringList)
  public
    destructor Destroy; override;
end;
Und hier der Destruktor:
Delphi-Quellcode:
destructor TStringTree.Destroy;
var
  i: Integer;
begin
  for i:=0 to Count-1 do begin
    if Objects [i] <> nil then begin
      if Objects [i] is TStringTree then
        TStringTree(Objects [i]).Free
      else
        Objects [i].Free;
      Objects [i] := nil;
    end;
  end;

  inherited;
end;
Approved? :)

lg Caps

Uwe Raabe 16. Jun 2020 11:31

AW: Rekursives Freigeben einer TStringList // Compilerfehler
 
Das geht auch einfacher:
Delphi-Quellcode:
destructor TStringTree.Destroy;
var
  i: Integer;
begin
  for i:=0 to Count-1 do begin
    Objects [i].Free;
    Objects [i] := nil;
  end;

  inherited;
end;


Alle Zeitangaben in WEZ +1. Es ist jetzt 22:10 Uhr.
Seite 2 von 4     12 34      

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