AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Zurück Delphi-PRAXiS Programmierung allgemein Algorithmen, Datenstrukturen und Klassendesign Delphi Unterschiedliche Objekte in einer Liste speichern
Thema durchsuchen
Ansicht
Themen-Optionen

Unterschiedliche Objekte in einer Liste speichern

Ein Thema von TUX_der_Pinguin · begonnen am 12. Feb 2016 · letzter Beitrag vom 15. Feb 2016
Antwort Antwort
Seite 1 von 2  1 2      
TUX_der_Pinguin

Registriert seit: 1. Jun 2005
Ort: Anholt (NRW)
609 Beiträge
 
Delphi 11 Alexandria
 
#1

Unterschiedliche Objekte in einer Liste speichern

  Alt 12. Feb 2016, 11:40
Hallo,

ich versuche mich gerade an einer Liste die unterschiedliche Objekte enthalten soll. Zur bessern Erklärung habe ich ein einfaches Beispiel erstellt.
Das Ziel ist ein Objekt zu erhalten welches eine Liste enthält die unterschiedliche Objekte verwalten kann. Dieses Objekt ist vom Typ "TItems" und enthält eine TObjectList in der Objekte vom Typ "ItemA" oder "ItemB" gespeichert werden können.

Ich stehe gerade auf dem Schlauch und weiß nicht was ich machen muss damit es funktioniert oder ob ich bereits bei dem ganzen Ansatz falsch liege.

Delphi-Quellcode:
unit Unit1;

interface

uses
  Classes,
  Contnrs;

type

  TField = class(TObject)
  private
    FName: String;
    FValue: String;
  public
    constructor Create;
    destructor Destroy; override;
    property Name: String read FName write FName;
    property Value: String read FValue write FValue;
  end;{TField}

  TFields = class(TObject)
  private
    FInnerList: TList;
    function GetItem(Index: Integer): TField;
    procedure SetItem(Index: Integer; Item: TField);
    function GetCount: Integer;
  public
    constructor Create;
    destructor Destroy; override;
    function Add(Item: TField): Integer;
    procedure Clear;
    procedure Delete(Index: Integer);
    property Count: Integer read GetCount;
    property Items[Index: Integer]: TField read GetItem write SetItem;
  end;{TFields}

  TItemA = class(TObject)
  private
    FName: String;
    FDesc: String;
    FFields: TFields;
  public
    constructor Create;
    destructor Destroy; override;
    property Name: String read FName write FName;
    property Desc: String read FDesc write FDesc;
    property Fields: TFields read FFields write FFields;
  end;{TItemA}

  TItemB = class(TObject)
  private
    FName: String;
    FComment: String;
    FText: String;
    FFields: TFields;
    FComp: TFields;
  public
    constructor Create;
    destructor Destroy; override;
    property Name: String read FName write FName;
    property Comment: String read FComment write FComment;
    property Text: String read FText write FText;
    property Fields: TFields read FFields write FFields;
    property Comp: TFields read FComp write FComp;
  end;{TItemB}

  TItems = class(TObject)
  private
    FInnerList: TObjectList;
    function GetData(Index: Integer): TObject;
    procedure SetData(Index: Integer; Data: TObject);
    function GetDataCount: Integer;
  public
    constructor Create;
    destructor Destroy; override;
    function Add(Data: TObject): Integer;
    procedure Clear;
    procedure Delete(Index: Integer);
    property Count: Integer read GetDataCount;
    property Items[Index: Integer]: TObject read GetData write SetData;
  end;{TItems}

implementation

uses
  SysUtils;

{ TField }

constructor TField.Create;
begin
  inherited;

  //init
  FName := EmptyStr;
  FValue := EmptyStr;
end;

destructor TField.Destroy;
begin
  //deinit
  FValue := EmptyStr;
  FName := EmptyStr;

  inherited;
end;

{ TFields }

function TFields.Add(Item: TField): Integer;
begin
  result := FInnerList.Add(Item);
end;

procedure TFields.Clear;
var
  Idx: Integer;

begin
  //deinit
  if Assigned(FInnerList) and (FInnerList.Count > 0) then
  begin
    for Idx := 0 to FInnerList.Count - 1 do
    begin
      TObject(FInnerList.Items[Idx]).Free;
    end;{for}
  end;{if}
  FInnerList.Clear;
end;

constructor TFields.Create;
begin
  inherited;

  //init
  FInnerList := TList.Create;
end;

procedure TFields.Delete(Index: Integer);
begin
  //Objekt freigeben
  TObject(FInnerList.Items[Index]).Free;

  //Eintrag löschen
  FInnerList.Delete(Index);
end;

destructor TFields.Destroy;
var
  Idx: Integer;

begin
  //deinit
  if Assigned(FInnerList) and (FInnerList.Count > 0) then
  begin
    for Idx := 0 to FInnerList.Count - 1 do
    begin
      TObject(FInnerList.Items[Idx]).Free;
    end;{for}
  end;{if}
  FInnerList.Free;

  inherited;
end;

function TFields.GetCount: Integer;
begin
  result := FInnerList.Count;
end;

function TFields.GetItem(Index: Integer): TField;
begin
  result := FInnerList.Items[Index];
end;

procedure TFields.SetItem(Index: Integer; Item: TField);
begin
  if Assigned(Item) then FInnerList.Items[Index] := Item;
end;

{ TItems }

function TItems.Add(Data: TObject): Integer;
begin
  result := FInnerList.Add(Data);
end;

procedure TItems.Clear;
var
  Idx: Integer;

begin
  //deinit
  if Assigned(FInnerList) and (FInnerList.Count > 0) then
  begin
    for Idx := 0 to FInnerList.Count - 1 do
    begin
      TObject(FInnerList.Items[Idx]).Free;
    end;{for}
  end;{if}
  FInnerList.Clear;
end;

constructor TItems.Create;
begin
  inherited;

  //init
  FInnerList := TObjectList.Create;
end;

procedure TItems.Delete(Index: Integer);
begin
  //Objekt freigeben
  TObject(FInnerList.Items[Index]).Free;

  //Eintrag löschen
  FInnerList.Delete(Index);
end;

destructor TItems.Destroy;
var
  Idx: Integer;

begin
  //deinit
  if Assigned(FInnerList) and (FInnerList.Count > 0) then
  begin
    for Idx := 0 to FInnerList.Count - 1 do
    begin
      TObject(FInnerList.Items[Idx]).Free;
    end;{for}
  end;{if}
  FInnerList.Free;

  inherited;
end;

function TItems.GetData(Index: Integer): TObject;
begin
  result := FInnerList.Items[Index];
end;

function TItems.GetDataCount: Integer;
begin
  result := FInnerList.Count;
end;

procedure TItems.SetData(Index: Integer; Data: TObject);
begin
  if Assigned(Data) then FInnerList.Items[Index] := Data;
end;

{ TItemA }

constructor TItemA.Create;
begin
  inherited;

  //init
  FName := EmptyStr;
  FDesc := EmptyStr;
  FFields := TFields.Create;
end;

destructor TItemA.Destroy;
begin
  //deinit
  FFields.Free;
  FDesc := EmptyStr;
  FName := EmptyStr;

  inherited;
end;

{ TItemB }

constructor TItemB.Create;
begin
  inherited;

  //init
  FName := EmptyStr;
  FComment := EmptyStr;
  FText := EmptyStr;
  FFields := TFields.Create;
  FComp := TFields.Create;
end;

destructor TItemB.Destroy;
begin
  //deinit
  FComp.Free;
  FFields.Free;
  FText := EmptyStr;
  FComment := EmptyStr;
  FName := EmptyStr;

  inherited;
end;

end.
Jedoch klappt das nicht und beim Versuch eines der Objekte in die Liste zu speichern erhalte ich nur eine Fehlermeldung "Ungültige Zeigeroperation"

Delphi-Quellcode:
procedure TForm17.Button1Click(Sender: TObject);
var
  Items: TItems;
  ItemA: TItemA;
  ItemB: TItemB;
  Field: TField;
  Idx: Integer;
  Comp: TField;

begin
  Items := TITems.Create;
  try
    //'Item A' erzeugen
    ItemA := TItemA.Create;
    ItemA.Name := 'test';
    ItemA.Desc := 'bla';

    for Idx := 1 to 10 do
    begin
      Field := TField.Create;
      Field.Name := 'Feld'+IntToStr(Idx);
      Field.Value := IntToStr(Idx);
      ItemA.Fields.Add(Field);
    end;{for}

    //'Item A' der Liste hinzufügen
    Items.Add(ItemA); //<- Ungültige Zeigeroperation


    //'Item B' erzeugen
    ItemB := TItemB.Create;
    ItemB.Name := 'BBB';
    ItemB.Comment := 'Zweiter Objekt-Typ';
    ItemB.Text := 'Bla Bla Mr. Freeman';

    for Idx := 1 to 10 do
    begin
      Field := TField.Create;
      Field.Name := 'Feld'+IntToStr(Idx);
      Field.Value := IntToStr(Idx);
      ItemB.Fields.Add(Field);
    end;

    for Idx := 1 to 5 do
    begin
      Comp := TField.Create;
      Comp.Name := 'Comp'+IntToStr(Idx);
      Comp.Value := IntToStr(Idx);
      ItemB.Comp.Add(Comp);
    end;

    //'Item B' der Liste hinzufügen
    Items.Add(ItemB);

  finally
    Items.Free;
  end;{try..finally}

end;
  Mit Zitat antworten Zitat
Benutzerbild von Neutral General
Neutral General

Registriert seit: 16. Jan 2004
Ort: Bendorf
5.219 Beiträge
 
Delphi 10.2 Tokyo Professional
 
#2

AW: Unterschiedliche Objekte in einer Liste speichern

  Alt 12. Feb 2016, 11:43
Hast du denn mal debugged?
Wo genau krachts denn im TItems.Add?
Ist die Liste erstellt? Wird der constructor von TItems (korrekt) ausgeführt?

Was mir noch aufgefallen ist: In TItems.Delete brauchst du folgende Zeile nicht (bzw. sie kann sogar Fehler verursachen):
TObject(FInnerList.Items[Index]).Free;
Eine TObjectList gibt die Objekte die entfernt werden automatisch vorher frei.
(Das wird durch die "OwnsObjects" Eigenschaft gesteuert, die auch im constructor optional übergeben werden kann)
Michael
"Programmers talk about software development on weekends, vacations, and over meals not because they lack imagination,
but because their imagination reveals worlds that others cannot see."

Geändert von Neutral General (12. Feb 2016 um 11:51 Uhr)
  Mit Zitat antworten Zitat
mkinzler
(Moderator)

Registriert seit: 9. Dez 2005
Ort: Heilbronn
39.858 Beiträge
 
Delphi 11 Alexandria
 
#3

AW: Unterschiedliche Objekte in einer Liste speichern

  Alt 12. Feb 2016, 11:45
Warum nimmst Du TList und nicht TObjectList?
Markus Kinzler
  Mit Zitat antworten Zitat
Benutzerbild von Sir Rufo
Sir Rufo

Registriert seit: 5. Jan 2005
Ort: Stadthagen
9.454 Beiträge
 
Delphi 10 Seattle Enterprise
 
#4

AW: Unterschiedliche Objekte in einer Liste speichern

  Alt 12. Feb 2016, 13:40
Warum nimmst Du TList und nicht TObjectList?
Er nimmt doch TObjectList ... und das verursacht das Problem. Denn er gibt die Instanz frei und die TObjectList will die Instanz auch noch freigeben.

Rumms!
Kaum macht man's richtig - schon funktioniert's
Zertifikat: Sir Rufo (Fingerprint: ‎ea 0a 4c 14 0d b6 3a a4 c1 c5 b9 dc 90 9d f0 e9 de 13 da 60)
  Mit Zitat antworten Zitat
Benutzerbild von Neutral General
Neutral General

Registriert seit: 16. Jan 2004
Ort: Bendorf
5.219 Beiträge
 
Delphi 10.2 Tokyo Professional
 
#5

AW: Unterschiedliche Objekte in einer Liste speichern

  Alt 12. Feb 2016, 13:53
Warum nimmst Du TList und nicht TObjectList?
Er nimmt doch TObjectList ... und das verursacht das Problem. Denn er gibt die Instanz frei und die TObjectList will die Instanz auch noch freigeben.

Rumms!
Er benutzt beides. TList bei TFields und TObjectList bei TItems
Michael
"Programmers talk about software development on weekends, vacations, and over meals not because they lack imagination,
but because their imagination reveals worlds that others cannot see."
  Mit Zitat antworten Zitat
Benutzerbild von Sir Rufo
Sir Rufo

Registriert seit: 5. Jan 2005
Ort: Stadthagen
9.454 Beiträge
 
Delphi 10 Seattle Enterprise
 
#6

AW: Unterschiedliche Objekte in einer Liste speichern

  Alt 12. Feb 2016, 14:02
Er benutzt beides. TList bei TFields und TObjectList bei TItems
Hab ich nicht gesehen ... also insgesamt etwas durcheinander der Aufbau
Kaum macht man's richtig - schon funktioniert's
Zertifikat: Sir Rufo (Fingerprint: ‎ea 0a 4c 14 0d b6 3a a4 c1 c5 b9 dc 90 9d f0 e9 de 13 da 60)
  Mit Zitat antworten Zitat
hoika

Registriert seit: 5. Jul 2006
Ort: Magdeburg
8.275 Beiträge
 
Delphi 10.4 Sydney
 
#7

AW: Unterschiedliche Objekte in einer Liste speichern

  Alt 12. Feb 2016, 14:21
Hallo,

was passiert, wenn du nur das hier machst?

Delphi-Quellcode:
ItemA := TItemA.Create;
Items.Add(ItemA); //<- Ungültige Zeigeroperation

Übrigens hat TObjectList ein OwnsObject (Boolean),
setze das auf False, dann kannst du das Free auch selbst machen.


Heiko
Heiko
  Mit Zitat antworten Zitat
Benutzerbild von Sir Rufo
Sir Rufo

Registriert seit: 5. Jan 2005
Ort: Stadthagen
9.454 Beiträge
 
Delphi 10 Seattle Enterprise
 
#8

AW: Unterschiedliche Objekte in einer Liste speichern

  Alt 12. Feb 2016, 14:32
Übrigens hat TObjectList ein OwnsObject (Boolean),
setze das auf False, dann kannst du das Free auch selbst machen.
Oder man lässt es auf true und spart sich jede Menge Boilerplate-Code
Kaum macht man's richtig - schon funktioniert's
Zertifikat: Sir Rufo (Fingerprint: ‎ea 0a 4c 14 0d b6 3a a4 c1 c5 b9 dc 90 9d f0 e9 de 13 da 60)
  Mit Zitat antworten Zitat
TUX_der_Pinguin

Registriert seit: 1. Jun 2005
Ort: Anholt (NRW)
609 Beiträge
 
Delphi 11 Alexandria
 
#9

AW: Unterschiedliche Objekte in einer Liste speichern

  Alt 15. Feb 2016, 08:41
Guten Morgen,

ihr hattet alle recht es lag daran das ich in der Klasse in der ich TObjectList verwendet habe dieses selbst freigeben wollte und als das Objekt sich selbst freigeben wollte knallte es. Ich stand da wohl irgendwie auf dem Schlauch, aber jetzt scheint alles zu funktionieren.

Noch mal vielen Dank euch allen und einen schönen Tag.
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

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

AW: Unterschiedliche Objekte in einer Liste speichern

  Alt 15. Feb 2016, 10:24
Auch bei OwnsObjects=True kann man Free selber machen.

Delphi-Referenz durchsuchenTObjectList.Delete <> Delphi-Referenz durchsuchenTObjectList.Remove
Neuste Erkenntnis:
Seit Pos einen dritten Parameter hat,
wird PoSex im Delphi viel seltener praktiziert.
  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 05:21 Uhr.
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