Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Delphi TPersistent-Listen in Stream schreiben und wieder lesen (https://www.delphipraxis.net/110928-tpersistent-listen-stream-schreiben-und-wieder-lesen.html)

Panthrax 26. Mär 2008 23:32


TPersistent-Listen in Stream schreiben und wieder lesen
 
Liste der Anhänge anzeigen (Anzahl: 1)
Hallo.

Ich möchte Komponenten und Listen (mit TPersistent-Nachfahren) in einem Stream speichern. Dieser soll dann mit ObjectBinaryToText und ObjectTextToBinary editierbar gemacht werden. Mit Komponenten funktioniert es zufriedenstellend; mit Listen nicht.

Zunächst habe ich mit einer Komponente mit Listen-Eigenschaft folgendes erfolgreich probiert:
  • Objekt erzeugen -> binär ablegen -> Objekt zerstören; mit Stream.WriteComponent, Component.DefineProperty
  • Objekt aus dem Binär-Stream wiederherstellen; mit Stream.ReadComponent, Component.DefineProperty
Wenn ich nun zwischendurch aus dem Binär-Stream einen Text-Stream machen will, kommt nur Murks heraus (siehe Bild). Für Komponenten (TComponent) und Kollektionen (TCollection und TCollectionItem) sind die Methoden WriteComponent und WriteCollection vorgesehen. Für Listen gibt es soetwas nicht (?).

Ein bisschen Quelltext:
Delphi-Quellcode:
procedure TMyComponent.WriteProp(Writer: TWriter);
var
  Component: TComponent;
begin
  Component:=TComponent.Create(nil);
  Component.Name:='Komponente1';
  Writer.WriteStr('MyProp');
  Writer.WriteListBegin;
  Writer.WriteComponent(Component);
  Writer.WriteListEnd;
  Component.Free;
end;
Für die Listen funktioniert zwar die Binär-Variante aber die Umwandlung zur Text-Variante ist nicht möglich, weil ich nicht weiß, wie die Binär-Variante strukturiert sein muss, dass sie ebenfalls in die Text-Variante umgewandelt werden kann (wenn das überhaupt möglich ist).

Mit Component.DefineBinaryProperty ist hier nicht geholfen, da es um die Editierbarkeit geht. Auch eine Kollektion hilft hier nicht weiter, da sie nicht das ist,was im Programm benötigt wird (größte Mankos: Keine Add(...)- und Extract-Methoden und kein Hinzufügen unterschiedlicher Objekte einer gleichen Basisklasse).

Meine Lösungsansatz führt nicht weiter. Irgendwelche Ideen?

shmia 27. Mär 2008 16:37

Re: TPersistent-Listen in Stream schreiben und wieder lesen
 
Zitat:

Zitat von Panthrax
Ich möchte Komponenten und Listen (mit TPersistent-Nachfahren) in einem Stream speichern. Dieser soll dann mit ObjectBinaryToText und ObjectTextToBinary editierbar gemacht werden. Mit Komponenten funktioniert es zufriedenstellend; mit Listen nicht.

Statt Listen (also TList Nachfahren) könntest du von TCollection ableiten.
Die Elemente der Collection müssen dann von TCollectionItem anstatt von TPersistent abgeleitet werden.
Die VCL sorgt dann dafür, dass die Collection und ihre Items in einem Stream geladen/gepseichert werden können.

Panthrax 27. Mär 2008 20:05

Re: TPersistent-Listen in Stream schreiben und wieder lesen
 
Hm, wie kann kann ich denn eine TCollection in eine Liste verwandeln, ohne dabei ihre Streaming-Funktionalität zu verlieren? Es gibt bei einer Kollektion keine Möglichkeit ein bestehendes Objekt hinzuzufügen (wie z.B. TComponentList.Add(TComponent.Create(nil))) und un es würden die Typem beim Speichern verloren gehen. Bisher enthalten die Listen Objekte unterschiedlicher Klassen (gemeinsame Basisklasse). Codeänderungen wären auderdem die Folge, denn die bisherigen Listen müssten dur die "TCollection-Liste" und die Elemente durch die "TCollectionItem-Elemente" ersetzt werden. Das würde das Problem nur verlagern - wohlmöglich auch weitere aufwerfen.

Ich versuche immernoch etwas wie Stream.WritePersistentList(const List: TPersistentList) zu implementieren. Es muss doch möglich sein, die Binärdaten so strukturiert abzulegen, dass sie von ObjectBinaryToText und ObjectTextToBinary zum Editieren umgewandelt werden können? Etwa so:

Delphi-Quellcode:
object MeineKomponente: TMyComponent
  MyInteger = 1                       // ein Integer
  MyString = 'Hallo Welt!'           // ein String
  MyCollection = <                   // eine Kollektion
    item
       ...
    end
    item
       ...
    end
    ...>
  MyList = (                          // eine Liste
    object Objekt1: TClass1
      ...
    end
    object Objekt2: TClass2
      ...
    end
    ...)
  ...
end
Wenn jemand eine ganz andere Lösung zum Streamen von Listen mit Elementen unterschiedlicher Klassen kennt, die sonst dem Delphi-eigenen Streaming gleichwertig ist, raus damit!


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