![]() |
Clear von TList überschreiben mit Objekten
Ich habe eine Klasse von TObject abgeleitet mit einer inneren Liste vom Typ TList. Diese fülle ich mit Objekten. Hin und wieder muss ich eine Kopie dieser Liste anfertigen. Dazu hab eich mir eine Methode Assign geschrieben:
Delphi-Quellcode:
Die Instanz für die Kopie ist global (in der Klasse), damit ich aber jetzt keine Speicherlecks bekomme rufe ich die Methode Clear auf bevor ich Klasse mit der LÖiste kopiere:
procedure TPageCollection.Assign(Source: TPageCollection);
var i : Integer; j : Integer; hobj : Integer; hObjSource : Integer; Page : TImageEnVect; begin // Klassenattribute kopieren PageParent := Source.PageParent; ScaleFactor := Source.ScaleFactor; for i := 0 to Source.Count - 1 do begin Page := TImageEnVect.Create(nil); // Seiteneigenschaften kopieren Page.Name := Source.Items[i].Name; Page.Parent := PageParent; //Source.Items[i].Parent; Page.Visible := False; Page.AllowOutOfBitmapMoving := False; Page.BorderStyle := bsNone; Page.ScrollBars := ssNone; Page.Cursor := crArrow; Page.DragMode := dmManual; Page.Width := Source.Items[i].Width; Page.Height := Source.Items[i].Height; Page.OnDragOver := Source.Items[i].OnDragOver; Page.OnDragDrop := Source.Items[i].OnDragDrop; Page.OnMouseDown := Source.Items[i].OnMouseDown; Page.OnObjectDblClick := Source.Items[i].OnObjectDblClick; Page.OnLayerNotify := Source.Items[i].OnLayerNotify; // Seiten-Layer kopieren. Bitmap nicht mit kopieren. for j := 1 to Source.Items[i].LayersCount - 1 do begin hobj := Page.LayersAdd; Page.Layers[hobj].Assign(Source.Items[i].Layers[j]); end; // Textobjekte for j := 0 to Source.Items[i].ObjectsCount - 1 do begin hobj := Page.AddNewObject; hObjSource := Source.Items[i].GetObjFromIndex(j); Page.ObjUserData[hObjSource] := Source.Items[i].ObjUserData[hObjSource]; Page.ObjKind[hobj] := Source.Items[i].ObjKind[hObjSource]; Page.ObjLeft[hobj] := Source.Items[i].ObjLeft[hObjSource]; Page.ObjTop[hobj] := Source.Items[i].ObjTop[hObjSource]; Page.ObjWidth[hobj] := Source.Items[i].ObjWidth[hObjSource]; Page.ObjHeight[hobj] := Source.Items[i].ObjHeight[hObjSource]; Page.ObjTextAlign[hobj] := Source.Items[i].ObjTextAlign[hObjSource]; Page.ObjFontName[hobj] := Source.Items[i].ObjFontName[hObjSource]; Page.ObjFontHeight[hobj] := Source.Items[i].ObjFontHeight[hObjSource]; Page.ObjPenColor[hobj] := Source.Items[i].ObjPenColor[hObjSource]; Page.ObjFontStyles[hobj] := Source.Items[i].ObjFontStyles[hObjSource]; Page.ObjText[hobj] := Source.Items[i].ObjText[hObjSource]; Page.Update; end; Add(Page); end; end;
Delphi-Quellcode:
Das sieht dann so aus:
procedure TPageCollection.Clear;
var i: Integer; begin for i := FInnerList.Count - 1 downto 0 do begin TObject(FInnerList.Items[i]).Free; end; // inherited; //FInnerList.Clear; end;
Delphi-Quellcode:
Nur leider bekomme ich eine AccessViolation in der Assign-Methode, wenn ich vorher meine Clear-Methode aufrufe.
TempPageCollection.Clear;
TempPageCollection.Assign(PageCollection); Was mache ich da falsch? Oder wie kann man sonst Speicherlecks vermeiden? |
Re: Clear von TList überschreiben mit Objekten
Wie ist den die Items-Eigenschaft von TPageCollection definiert ?
|
Re: Clear von TList überschreiben mit Objekten
Delphi-Quellcode:
Und dann die Property dazu:
function TPageCollection.GetItem(Index: Integer): TImageEnVect;
begin Result := FInnerList.Items[Index]; end; procedure TPageCollection.SetItem(Index: Integer; Item: TImageEnVect); begin FInnerList.Items[Index] := Item; end;
Delphi-Quellcode:
property Items[Index: Integer]: TImageEnVect read GetItem write SetItem;
|
Re: Clear von TList überschreiben mit Objekten
hm....also das einzige was mir jetzt auffählt wäre im Clear.
Müsste das nicht
Delphi-Quellcode:
statt
TImageEnVect(FInnerList.Items[i]).Free;
Delphi-Quellcode:
Ansonsten mal mit dem Debuger rein, und guggen wo er genau im Assign semmelt.
TObject(FInnerList.Items[i]).Free;
|
Re: Clear von TList überschreiben mit Objekten
ist FInnerList denn immer initialisiert ?
sonst greift Count ins Leere. |
Re: Clear von TList überschreiben mit Objekten
Die innere Liste wird im Konstruktor erzeugt und im Destruktor wieder freigegeben. Und die TempPageCollection existiert auch. Durch die Schleife in der Clear-Methode läuft er ja auch durch, nur in der darauffolgenden Assign-Methode kracht es dann und zwar wenn der betreffende Code zum zweiten mal aufgerufen wird. Aber ich kopiere ja, also dürfte die Clear-Methode ja nichts aus der original Liste löschen oder so.
|
Re: Clear von TList überschreiben mit Objekten
Ohne jetzt einen konkreten Fehler zu sehen, zwei etwas allgemeinere Tips:
* FreeAndNil statt Free verwenden; dadurch sieht man beim Debuggen wenigstens sofort, daß das Objekt vorher mal freigegeben wurde, statt einen Zeiger ins Irgendwo zu haben. * In Assign nochmal expliziter mit Self. arbeiten, für den Fall das sich da irgendwo Namen und Sichtbarkeiten überschneiden. |
Re: Clear von TList überschreiben mit Objekten
Öhm...du solltest vielleicht auch den entsprechenden Listeneintrag beim Clear löschen, nicht nur das Objekt freigeben :)
Ansonsten..wie ich oben geschrieben hab. |
Re: Clear von TList überschreiben mit Objekten
Hallo,
soll Clear die Liste leeren oder nur die Objekte freigeben, in ersterem Fall fehlt was
Delphi-Quellcode:
Durch Free des Objektes wird zwar as Objekt freigegeben,
procedure TPageCollection.Clear;
var i: Integer; begin for i := FInnerList.Count - 1 downto 0 do begin TObject(FInnerList.Items[i]).Free; // das fehlt FInnerList.Delete(i); // oder ungetestet // FInnerLIst.Remove(FInnerList.Items[i]) end; end; aber nicht der Listeneintrag (Pointer), Count bliebt also immer gleich, nur der Pointer zeugt ins Nirvana. Ich würde per {$IFDEF DEBUG} eh immer FreeAndNIL benutzen, dann bekommt du wenigstens (hoffentlich) ne bessere Fehlermeldung. Heiko PS: Schon mal TObjectlist versucht ? |
Re: Clear von TList überschreiben mit Objekten
@hoika: ein FInnerList.Clear reicht auch aus um alle Listeneinträge der TList zu entfernen, das hat Luckie auch gemacht (allerdings ist der code bei ihm auskommentiert).
Delphi-Quellcode:
procedure TPageCollection.Clear;
var i: Integer; begin for i := FInnerList.Count - 1 downto 0 do begin TObject(FInnerList.Items[i]).Free; end; FInnerList.Clear; end; |
Alle Zeitangaben in WEZ +1. Es ist jetzt 18:57 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