AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Zurück Delphi-PRAXiS Projekte himXML (gesprochen himix ML)

himXML (gesprochen himix ML)

Ein Thema von himitsu · begonnen am 12. Mär 2009 · letzter Beitrag vom 11. Nov 2020
Antwort Antwort
Seite 2 von 35     12 3412     Letzte » 
Benutzerbild von himitsu
himitsu
Registriert seit: 11. Okt 2003
wollt eigentlich nur mal die ersten Fortschritte eines vor kurzem längerem begonnen Projektes vorstellen

es ist "nur" eine weitere kleine XML-Klasse.

der Grund war eigentlich, daß mir andere XML-Projekte zu rießig sind
und MSMXL (Hier im Forum suchenTXMLDocument) nicht grad schnell.

[info]
aktuell entwickle ich unter D2009 und direkt kompatibel düfte es bis Delphi 2006 / Turbo Delphi sein

und Aufgrund einiger Anfragen versuch ich, sobald es da "gut" läuft, eine Extraversion weiter abwärtskompatibel zu machen ... mal sehn, bis wie weit runter das geht
[/info]
[add 04.01.2001] aktuell sieht es so aus, als wenn es schonmal bis zu D7 läuft [add/]

aktueller Post: http://www.delphipraxis.net/internal...116416#1116416 #193
  • aktuell ab Delphi 2006 bzw. Turbo Delphi und auch für Delphi2009
    ab Windows 2000 und öfters mal in Windows 7 getestet
  • zu den möglichen unterstützten Delphi-Versionen gibt es, Aufgrund häufiger Nachfragen, in Beitrag #191 eine kleine Zusammenfassung.
  • Lizenzen: MPL v1.1 , GPL v3.0 oder LGPL v3.0
    es sollte also hoffentlich mit vielen anderen Lizenzen kompatibel sein
  • arbeitet weitetgehend nach XML-Standard 1.0 (Fifth Edition)
    und Extensible Markup Language (XML) 1.1 (Second Edition)
  • ist nicht validierend
  • ein einzulesendes Dokument sollte aber möglichst "Well-Formed" (wohlgeformt) sein
  • es wird nur der "Standard" unterstützt ... andere DTDs oder ähnliche Informationen werden nicht unterstützt
  • also <!xyz text> wird zwar ausgelesen, aber "text" wird nicht auswertet
  • <![CDATA[text]]> wird dagen natürlich behandelt
  • <?xyz text ?> wird unterstützt, allerdings wird "text" geparst und in Parameter zerlegt (also wird es Probleme geben falls sich "text" mal nicht in Parameter zerlegen läßt)
    (nach XML 1.0 wird "text" eigentlich nicht zuerlegt im XML-Baum abgelegt, sondern nur XML-Prozessor-intern verarbeitet oder ignoriert)
  • unterstützte Kodierungen (teilweise auch via BOM):
    UTF-7, UTF-8, UTF-16, ISO-10646-UCS-2 (littel endian), Unicode (big endian)
    ISO-8859-1 bis ISO-8859-9, ISO-2022-JP, EUC-JP, SHIFT-JIS und WINDOWS-1250 bis WINDOWS-1258
  • UCS-4 (4 Byte-Unicode) wird nicht unterstützt,
    denn leider kann man immernoch nicht mit UCS4Char und UCS4String richtig arbeiten, vorallem da sie vollkommen inkompatibel zu den anderen Strings sind (bauen intern nur auf LongWord auf, statt auf einen "Char")
  • im Gegensatz zu MSXML (TXMLDocument) kann die Kodierung stärker umgeschaltet werden
    (auch von ANSI/UTF-8 nach Unicode ... MSXML kann z.B. nur umschalten, wenn sich die Zeichengröße in Byte nicht ändert)
  • arbeitet nur mit Klassen
    (eine Variante für sehr große Dateien und mit Interfaces aufgebaut, welche die Dateninhalte vorwiegend direkt in der Datei liest, speichert und ändert, kommt vielleicht irgendwann mal)
  • eine Thread-Absicherung wurde letztendlich doch nicht direkt integriert,
    aber die CriticalSection wurde dringelassen und kann direkt verwendet werden:
    Delphi-Quellcode:
    XML._Lock;
    Try
      ... mach was mit dem XML ...
    Finally
      XML._Unlock;
    End;
  • es muß nur jeweils eine Datei eingebunden werden
    himXML, himXMLi, himXML_UseDLL oder himXML_UseDLLi,
    jenachdem mit welcher Schnittstelle man arbeitet
    > wichtige Typen und Konstanten werden in den Dateien weitergeleitet, falls sie nicht in der entsprechenden Unit definiert sind
  • ein kleiner Benchmark ist aber schon lauffähig:
    Delphi-Quellcode:
    // inklusive Zeitmessung siehe FormCreate

    XML := TXMLDocument.Create(Self);
    XML.Active := True;
    XML.Version := '1.0';
    XML.StandAlone := 'yes';
    XML.Encoding := 'UTF-8';
    XML.Options := [doNodeAutoIndent];
    XML.AddChild('xml');
    For i := 0 to 100000 do XML.DocumentElement.AddChild('Node' + IntToStr(i));
    XML.SaveToFile('test2.xml');
    XML.Free;

    // versus

    XML := TXMLFile.Create(nil);
    For i := 0 to 100000 do XML.RootNode.Nodes.Add('Node' + IntToStr(i));
    XML.SaveToFile('test.xml');
    XML.Free;
    Delphi-Quellcode:
    //10.000 Nodes (ergibt ca. eine 150 KB-Datei)

    TXMLDocument = fill:6749 save:99
    TXMLFile = fill: 10 save: 7

    // 100.000 Nodes (ergibt ca. eine 1,5 MB-Datei)

    TXMLDocument = fill:708145 save:348 load:331
    TXMLFile = fill: 189 save:150 load:319
    Zeiten in Millisekunden

    weitere Tests und Vergleiche (auch mit anderen Libs), siehe demos\CheckLibs.dpr
    wer keine Lust hat, vorallem wegen MSXML 'ne halbe Stunde ein Programm laufen zu lassen ... auf den nächsten Seiten befinden sich sporatisch die Testergebnisse verteilt
  • "wichtige" Grundfunktionen von .RootNode wurden direkt in die Basisklasse (TXMLFile) umgelegt, sowie von .Nodes in dessen Mutterklasse (TXMLNode)

    .Node (.Nodes.Node[], .RootNode.Nodes.Node[] bzw .Nodes[], .RootNode.Nodes[]), .NodeNF (.Nodes.NodeNF bzw. .RootNode.Nodes.NodeNF), .NodeList (.Nodes.NodeList bzw. .RootNode.Nodes.NodeList), .NodeListNF (.Nodes.NodeListNF bzw. .RootNode.Nodes.NodeListNF) und AddNode (.Nodes.Add bzw. .RootNode.Nodes.Add)
    (in Groß siehe Beitrag #65)
  • Pfadangaben sind möglich
    in Kurz also XML['node1\node2'] statt XML.RootNode.Nodes['node1'].Nodes['node2']
    ebenso können Parameter und Indizes im Nodepfad angegeben werden (siehe Hilfesektion in himXMLi.pas)
  • es kann direkt eine Inline-Verschlüsselung genutzt werden
    einfach für den gewünschten Knoden zusweisen und alles via .Text und .Text_S wird automatisch beim Zuweisen/Auslesen ver-/entschlüsselt
    Delphi-Quellcode:
    Node.Crypted := '*'; // Standardverschlüsselung nutzen (erstes in der Liste)
    Node.Crypted := 'RC4'; // RC4 verwenden
    Node.Text := X; // zuweisen und automatisch verschlüsseln
    X := Node.Text; // automatisch entschlüsseln und auslesen
    If Node.Crypted <> 'Then // nur prüfen ob es verschlüsselt ist
    If Node.Crypted = 'RC4Then // prüfen ob es RC4 verschlüsselt ist

    XML.CryptData['RC4'] := S; // anderen Key für RC4 festlegen
    da die Verschlüsselungsroutine Zugriff auf die Node-Attribute hat, könnte man auch eine eigene Verschlüsselung zuweisen, welche z.B. über ein weiteres Attribut jeden Node mit einem eigenem Key verschlüsselt.
    Delphi-Quellcode:
    XML.Cryptors // listet alle registrierten Verschlüsselungen auf
    XML.SetCryptor // setzt/lösche eine Verschlüsselung
    XML.CryptProc // siehe SetCryptor
    XML.CryptData // siehe SetCryptor
    XML.CryptAttrName // der Name des Attributes, welcher die aktive Verschlüsselung enthält
    • himXML.pas
      - die ganzen Klassen
    • himXMLi.pas
      - die zugehörigen Interfaces (sie spiegeln auch die Funktionalität der Klassen wieder .. nur da halt über TXML... statt IXML...)
      - und enthält auch eine "kleine" Hilfe-Sektion
    • himXML_Tools.pas
      - kleine Zusatzmodule:
      TSimpleAssocVariantArray - sehr einfaches String-assoziatives Array of Variant
      TXMLSerializeRecord_Creator - erzeugt eine Record-Definition für .Serialize/.DeSerialize von Records
      TXMLIniFile - verwendet eine XML-Datei wie eine INI und läßt sich auch wie IniFiles.TIniFile nutzen
      TXMLRegistry - das Gleiche, nur halt mit Registry.TRegistry
      TXMLDatabase - noch unfertig, soll aber irgendwann mal eine XML-Datei wie eine kleine SQL-DB nutzbar machen
    • demos\CheckLibs.dpr
      - einige Tests/Benchmarks zum Projekt und anderen XML-Libs
    • demos\CheckLib_UseDLL.dpr und demos\CheckLib_UseDLLi.dpr
      - das Gleiche, nur über ein externes himXML via DLLs (siehe weiter unten)
    • demos\Demo_Serialize.dpr
      - zeigt einiges zu den (De)Serialisierungsfunktionsn
    • demos\Demo_Tree.dpr
      - 'ne kleine Demo, wo eine XML-Datei in eine TreeView geladen und angezeigt wird
    • demos\SAXParser.dpr
      - hier wird eine kleine XML-Testdatei sequentiel geladen
    • DLLs\himXML_UseDLL.pas und DLLs\himXML_UseDLLi.pas
      - dieses sind die Schnittstellen-Units für die DLLs
      einmal als Objekte und das Andere (mit dem i) als Interfaces
      bei der Objektversion sind die Exceptions noch nicht umgeleitet, also möglichst welche vermeiden
      in der Interfaceversion funktioniert das Freigeben nochn nicht
    • DLLs\himXML_DLL.dpr und DLLs\himXML_DLLi.dpr
      - die zugehörigen DLLs
    • other\*
      - einige andere XML-Libs (verwendet in demos\CheckLibs* )
    • test files\*
      - einige "XML"-Dateien zum Testen

ansonsten bin ich für Tipps und Vorschläge dankbar

[add 13.03.2009]
ach ja falls der Name jemandem nich gefällt ... Beschwerden bitte an Matze richten

[add]
im Beitrag #193 wird etwas über die im Download mit enthalenen anderen XML-Libs geschrieben.


[add]
Achtung, beim Forenupgrad der DP sind die Dateikommentare verschwunden und die Dateinamen sind unglücklich importiert wurden.

himxml_246.7z = v0.9 21.05.2009
himxml_164.7z = v0.99d 12.01.2010
other.7z = 30.12.2010 (only the "other" directory)
Angehängte Dateien
Dateityp: 7z himxml_246.7z (2,70 MB, 1135x aufgerufen)
Dateityp: 7z himxml_164.7z (2,35 MB, 938x aufgerufen)
Dateityp: 7z other.7z (1,61 MB, 320x aufgerufen)
Garbage Collector ... Delphianer erzeugen keinen Müll, also brauchen sie auch keinen Müllsucher.
my Delphi wish list : BugReports/FeatureRequests

Geändert von himitsu (30. Dez 2010 um 08:29 Uhr)
 
Benutzerbild von himitsu
himitsu

 
Delphi 12 Athens
 
#11
  Alt 2. Apr 2009, 09:27
bei diesem Code
Code:
Type TForm1 = Class(TForm)
    Label1: TLabel;
    Memo1: TMemo;
    Procedure FormCreate(Sender: TObject);
  Private
    _xyz: TMyProc;
    Procedure MyProc(x: Integer);
  Published
    Property xyz: TMyProc read _xyz write _xyz Stored True;
  End;


XML := TXMLFile.Create;

Node := XML.RootNode.Nodes.Add('node1');
Node.Attributes['attr1'] := '123';
Node.Attributes['attr2'] := '456';
Node.Nodes.Add('node1_2');
Node := Node.Nodes.Add('node1_3');
Node.Nodes.Add('node1_3_1');
Node := XML.RootNode.Nodes.Add('node2');
Node := Node.Nodes.Add('node2_1');
Node.Attributes['attr3'] := 'abc';

Form1.xyz := MyProc;
Node := Node.Nodes.Add('object');
Node.Serialize(Form1);

XML.SaveToFile('test.xml');
kommt dieses raus
XML-Code:
<?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
<xml>
  <node1 attr1="123" attr2="456">
    <node1_2 />
    <node1_3>
      <node1_3_1 />
    </node1_3>
  </node1>
  <node2>
    <node2_1 attr3="abc">
      <object>
        <Tag>0</Tag>
        <AlignWithMargins>False</AlignWithMargins>
        <Left>66</Left>
        <Top>72</Top>
        <Cursor>0</Cursor>
        <HelpType>htContext</HelpType>
        <HelpContext>0</HelpContext>
        <Margins ClassType="TMargins">
          <Left>3</Left>
          <Top>3</Top>
          <Right>3</Right>
          <Bottom>3</Bottom>
        </Margins>
        <ParentCustomHint>True</ParentCustomHint>
        <HorzScrollBar ClassType="TControlScrollBar">
          <ButtonSize>0</ButtonSize>
          <Color>clBtnHighlight</Color>
          <Increment>8</Increment>
          <Margin>0</Margin>
          <ParentColor>True</ParentColor>
          <Position>0</Position>
          <Range>0</Range>
          <Smooth>False</Smooth>
          <Size>0</Size>
          <Style>ssRegular</Style>
          <ThumbSize>0</ThumbSize>
          <Tracking>False</Tracking>
          <Visible>True</Visible>
        </HorzScrollBar>
        <VertScrollBar ClassType="TControlScrollBar">
          <ButtonSize>0</ButtonSize>
          <Color>clBtnHighlight</Color>
          <Increment>8</Increment>
          <Margin>0</Margin>
          <ParentColor>True</ParentColor>
          <Position>0</Position>
          <Range>0</Range>
          <Smooth>False</Smooth>
          <Size>0</Size>
          <Style>ssRegular</Style>
          <ThumbSize>0</ThumbSize>
          <Tracking>False</Tracking>
          <Visible>True</Visible>
        </VertScrollBar>
        <Align>alNone</Align>
        <AlphaBlend>False</AlphaBlend>
        <AlphaBlendValue>255</AlphaBlendValue>
        <AutoSize>False</AutoSize>
        <BorderIcons>[biSystemMenu,biMinimize,biMaximize]</BorderIcons>
        <BorderStyle>bsSizeable</BorderStyle>
        <BorderWidth>0</BorderWidth>
        <Caption>Form1</Caption>
        <ClientHeight>201</ClientHeight>
        <ClientWidth>329</ClientWidth>
        <Color>clBtnFace</Color>
        <TransparentColor>False</TransparentColor>
        <TransparentColorValue>clBlack</TransparentColorValue>
        <Constraints ClassType="TSizeConstraints">
          <MaxHeight>0</MaxHeight>
          <MaxWidth>0</MaxWidth>
          <MinHeight>0</MinHeight>
          <MinWidth>0</MinWidth>
        </Constraints>
        <Ctl3D>True</Ctl3D>
        <UseDockManager>False</UseDockManager>
        <DefaultMonitor>dmActiveForm</DefaultMonitor>
        <DockSite>False</DockSite>
        <DoubleBuffered>False</DoubleBuffered>
        <DragKind>dkDrag</DragKind>
        <DragMode>dmManual</DragMode>
        <Enabled>True</Enabled>
        <ParentFont>False</ParentFont>
        <Font ClassType="TFont">
          <Charset>1</Charset>
          <Color>clWindowText</Color>
          <Height>-11</Height>
          <Name>Tahoma</Name>
          <Orientation>0</Orientation>
          <Pitch>fpDefault</Pitch>
          <Style>[]</Style>
        </Font>
        <FormStyle>fsNormal</FormStyle>
        <GlassFrame ClassType="TGlassFrame">
          <Enabled>False</Enabled>
          <Left>0</Left>
          <Top>0</Top>
          <Right>0</Right>
          <Bottom>0</Bottom>
          <SheetOfGlass>False</SheetOfGlass>
        </GlassFrame>
        <KeyPreview>False</KeyPreview>
        <Padding ClassType="TPadding">
          <Left>0</Left>
          <Top>0</Top>
          <Right>0</Right>
          <Bottom>0</Bottom>
        </Padding>[list=1]False</OldCreateOrder>
        <ParentBiDiMode>True</ParentBiDiMode>
        <PopupMode>pmNone</PopupMode>
        <Position>poDefaultPosOnly</Position>
        <PrintScale>poProportional</PrintScale>
        <Scaled>True</Scaled>
        <ScreenSnap>False</ScreenSnap>
        <SnapBuffer>10</SnapBuffer>
        <Visible>False</Visible>
        <WindowState>wsNormal</WindowState>
        <OnCreate>TForm1:Form1:$0047BB54</OnCreate>
        <xyz>TForm1:Form1:$0047B894</xyz>
        <Components>
          <Component ClassType="TLabel">
            <Tag>0</Tag>
            <AlignWithMargins>False</AlignWithMargins>
            <Left>32</Left>
            <Top>37</Top>
            <Width>31</Width>
            <Height>13</Height>
            <Cursor>0</Cursor>
            <HelpType>htContext</HelpType>
            <HelpContext>0</HelpContext>
            <Margins ClassType="TMargins">
              <Left>3</Left>
              <Top>3</Top>
              <Right>3</Right>
              <Bottom>3</Bottom>
            </Margins>
            <ParentCustomHint>True</ParentCustomHint>
            <Align>alNone</Align>
            <Alignment>taLeftJustify</Alignment>
            <AutoSize>True</AutoSize>
            <Caption>Label1</Caption>
            <Constraints ClassType="TSizeConstraints">
              <MaxHeight>0</MaxHeight>
              <MaxWidth>0</MaxWidth>
              <MinHeight>0</MinHeight>
              <MinWidth>0</MinWidth>
            </Constraints>
            <DragCursor>-12</DragCursor>
            <DragKind>dkDrag</DragKind>
            <DragMode>dmManual</DragMode>
            <EllipsisPosition>epNone</EllipsisPosition>
            <Enabled>True</Enabled>
            <GlowSize>0</GlowSize>
            <ParentBiDiMode>True</ParentBiDiMode>
            <ParentColor>True</ParentColor>
            <ParentFont>True</ParentFont>
            <ParentShowHint>True</ParentShowHint>
            <ShowAccelChar>True</ShowAccelChar>
            <Layout>tlTop</Layout>
            <Visible>True</Visible>
            <WordWrap>False</WordWrap>
          </Component>
          <Component ClassType="TMemo">
            <Tag>0</Tag>
            <AlignWithMargins>False</AlignWithMargins>
            <Left>32</Left>
            <Top>56</Top>
            <Width>257</Width>
            <Height>89</Height>
            <Cursor>0</Cursor>
            <HelpType>htContext</HelpType>
            <HelpContext>0</HelpContext>
            <Margins ClassType="TMargins">
              <Left>3</Left>
              <Top>3</Top>
              <Right>3</Right>
              <Bottom>3</Bottom>
            </Margins>
            <ParentCustomHint>True</ParentCustomHint>
            <TabStop>True</TabStop>
            <Align>alNone</Align>
            <Alignment>taLeftJustify</Alignment>
            <BevelEdges>[beLeft,beTop,beRight,beBottom]</BevelEdges>
            <BevelInner>bvRaised</BevelInner>
            <BevelKind>bkNone</BevelKind>
            <BevelOuter>bvLowered</BevelOuter>
            <BorderStyle>bsSingle</BorderStyle>
            <CharCase>ecNormal</CharCase>
            <Color>clWindow</Color>
            <Constraints ClassType="TSizeConstraints">
              <MaxHeight>0</MaxHeight>
              <MaxWidth>0</MaxWidth>
              <MinHeight>0</MinHeight>
              <MinWidth>0</MinWidth>
            </Constraints>
            <DragCursor>-12</DragCursor>
            <DragKind>dkDrag</DragKind>
            <DragMode>dmManual</DragMode>
            <Enabled>True</Enabled>
            <HideSelection>True</HideSelection>
            <ImeMode>imDontCare</ImeMode>
            <Lines ClassType="TMemoStrings">
              <Strings>
                <String>Memo1</String>
                <String />
                <String>abc</String>
              </Strings>
            </Lines>
            <MaxLength>0</MaxLength>
            <OEMConvert>False</OEMConvert>
            <ParentBiDiMode>True</ParentBiDiMode>
            <ParentColor>False</ParentColor>
            <ParentCtl3D>True</ParentCtl3D>
            <ParentDoubleBuffered>True</ParentDoubleBuffered>
            <ParentFont>True</ParentFont>
            <ParentShowHint>True</ParentShowHint>
            <ReadOnly>False</ReadOnly>
            <ScrollBars>ssNone</ScrollBars>
            <TabOrder>0</TabOrder>
            <Visible>True</Visible>
            <WantReturns>True</WantReturns>
            <WantTabs>False</WantTabs>
            <WordWrap>True</WordWrap>
          </Component>
        </Components>
      </object>
    </node2_1>
  </node2>
</xml>
isses OK so?
(hier sind jetzt nur Stored-Property <> Default-Wert drin)



bei Propertys, welche nicht von meiner Serialize-Prozedur verarbeitet werden, wird erstmal soeine Prozedur aufzurufen (wenn angegeben) und wenn Beides das Property nicht verarbeitet, dann gibt's eine Exception.
Delphi-Quellcode:
Function SerializeProc(C: TObject; Const PropertyName: AnsiString; NodeList: TXMLNodeList): Boolean;
  Var Node: TXMLNode;

  Begin
    Node := NodeList.Add(PropertyName);
    Node.Attributes['unknown'] := 'True';
    //Node.Attributes['Name'] := GetPropInfo(C, PropertyName).PropType^.Name;
    //Node.Attributes['Type'] := cTypeKind[GetPropInfo(C, PropertyName).PropType^.Kind];
    Result := True; // True wenn verarbeitet ... egal ob ein Node erstellt, oder ignoriert wurde
      // bei False wird (womöglich) eine Exception geworfen, daß ein Property nicht bearbeitet wurde
  End;

Node.Serialize(Form1, [], SerializeProc);
man kann diese Prozedut auch nutzen, um selbst anzugeben, welche Properties gespeichert bzw. geladen (bei Deserialize) werden sollen.
Ist NodeList = nil, dann wird über Result bestimmt, was geschehen soll.
True = Property versuchen zu speichern (falls unbekannt, dann wird diese Funktion nochmals aufgerufen ... siehe vorheriges Beispiel)
False = Property nicht speichern/laden
Delphi-Quellcode:
Function SerializeProc(C: TObject; Const PropertyName: AnsiString; NodeList: TXMLNodeList): Boolean;
  Var Node: TXMLNode;

  Begin
    If Assigned(NodeList) Then Begin
      Node := NodeList.Add(PropertyName);
      Node.Attributes['unknown'] := 'True';
      //Node.Attributes['Name'] := GetPropInfo(C, PropertyName).PropType^.Name;
      //Node.Attributes['Type'] := cTypeKind[GetPropInfo(C, PropertyName).PropType^.Kind];
      Result := True;
    End Else Begin
      // only if xsQueryBefore is set
      Result := PropertyName <> 'Name';
    End;
  End;

Node.Serialize(Form1, [xsNonStoredProperties, xsQueryBefore], SerializeProc);

das Ganze läßt sich dann natürlich auch noch auf bestimmte Property eingrenzen:
Delphi-Quellcode:
Function SerializeProc(C: TObject; Const PropertyName: AnsiString; NodeList: TXMLNodeList): Boolean;
  Var Node: TXMLNode;

  Begin
    If (C is TMyClass) and (PropertyName = 'MyProperty') Then Begin
      Node := NodeList.Add(PropertyName);
      Node.Data := LeseDaten(C, PropertyName);
      Result := True;
    End Else Result := False;
  End;

// hier wird nur das Property "MyProperty" des übergebenen Objektes gespeichert

Function SerializeProc(C: TObject; Const PropertyName: AnsiString; NodeList: TXMLNodeList): Boolean;
  Var Node: TXMLNode;

  Begin
    If Assigned(NodeList) Then Begin
      If (C is TMyClass) and (PropertyName = 'MyProperty') Then Begin
        Node := NodeList.Add(PropertyName);
        Node.Data := LeseDaten(C, PropertyName);
        Result := True;
      End Else Result := False;
    End Else Begin
      // only if xsQueryBefore is set
      Result := (C is TMyClass) and (PropertyName = 'MyProperty');
    End;
  End;

// hier wird alles gespeichert, was meine Funktion speichern kann
// und zusätzlich noch das Property "MyProperty" (welches meine Funktion nicht kennt)

Function SerializeProc(C: TObject; Const PropertyName: AnsiString; NodeList: TXMLNodeList): Boolean;
  Var Node: TXMLNode;

  Begin
    If Assigned(NodeList) Then Begin
      If (C is TMyClass) and (PropertyName = 'MyProperty') Then Begin
        Node := NodeList.Add(PropertyName);
        Node.Data := LeseDaten(C, PropertyName);
        Result := True;
      End Else Result := True;
    End Else Begin
      // only if xsQueryBefore is set
      Result := True;
    End;
  End;
als Parameter gibt's dieses:
Code:
xsSortProperties      sortiert die Properties
xsDefaultProperties   speichert auch Properties, welche ihrem "Default"-Wert entsprechen
xsNonStoredProperties speichert auch Properties, welche nicht mit "Stored" markiert sind
xsSaveClassType       speichert den Klassen-Typ
                         siehe <Lines ClassType="TMemoStrings"> wäre es mit angegeben gewesen,
                         dann stünde statt <object> jetzt <object ClassType="TForm1">
xsSavePropertyInfos   ist mehr für Debugzwecke
                         entspricht Name=".PropType^.Name" Type=".PropType^.Kind";
xsQueryBefore         ruft SerializeProc auf und fragt, ob das Property gespeichert
                         werden soll ... siehe Beispiele
ich räum jetzt noch den Code etwas auf
und lad dann eventuell den aktuellen Code mal hoch
ansonsten bastel jetzt erstmal wieder am Parsen rum (das Lesen geht immernoch nicht so, wie ich es gern hätte)




[add]
im vollen Modus kommt sowas raus
Delphi-Quellcode:
Function SerializeProc(C: TObject; Const PropertyName: AnsiString; NodeList: TXMLNodeList): Boolean;
  Const cTypeKind: Array[TTypeKind] of String = ('Unknown', 'Integer', 'Char', 'Enumeration',
     'Float', 'String', 'Set', 'Class', 'Method', 'WChar', 'LString', 'WString',
     'Variant', 'Array', 'Record', 'Interface', 'Int64', 'DynArray', 'UString');

  Var Node: TXMLNode;

  Begin
   If Assigned(NodeList) Then Begin
     Node := NodeList.Add(PropertyName);
     Node.Attributes['unknown'] := 'True';
     Node.Attributes['unknown_Name'] := GetPropInfo(C, PropertyName).PropType^.Name;
     Node.Attributes['unknown_Type'] := cTypeKind[GetPropInfo(C, PropertyName).PropType^.Kind];
     Result := True;
   End Else Begin
     // only if xsQueryBefore is set
     Result := PropertyName <> 'Name';
   End;
  End;

Node.Serialize(Form1, [xsDefaultProperties, xsNonStoredProperties, xsSaveClassType,
  xsSavePropertyInfos, xsQueryBefore], SerializeProc);
es wird alles gespeichert ... nur "Name" nicht ... und das nicht behandelbare TIcon wurde von SerializeProc erstellt
XML-Code:
<?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
<xml>
  ...
  <node2>
   <node2_1 attr3="abc">
     <object ClassType="TForm1">
      <Tag Name="Integer" Type="Integer">0</Tag>
      <AlignWithMargins Name="Boolean" Type="Enumeration">False</AlignWithMargins>
      <Left Name="Integer" Type="Integer">88</Left>
      <Top Name="Integer" Type="Integer">96</Top>
      <Width Name="Integer" Type="Integer">337</Width>
      <Height Name="Integer" Type="Integer">230</Height>
      <Cursor Name="TCursor" Type="Integer">0</Cursor>
      <Hint Name="string" Type="UString" />
      <HelpType Name="THelpType" Type="Enumeration">htContext</HelpType>
      <HelpKeyword Name="string" Type="UString" />
      <HelpContext Name="THelpContext" Type="Integer">0</HelpContext>
      <Margins Name="TMargins" Type="Class" ClassType="TMargins">
        <Left Name="TMarginSize" Type="Integer">3</Left>
        <Top Name="TMarginSize" Type="Integer">3</Top>
        <Right Name="TMarginSize" Type="Integer">3</Right>

      ...

      <Icon unknown="True" unknown_Name="TIcon" unknown_Type="Class" />
Das Deserialize fehlt auch noch ... mach erstmal Serialize fertig und wende mich dann dem deserialisieren zu (hab da noch ein paar Problemchen beim Speichern auszumerzen)
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

 
Delphi 12 Athens
 
#12
  Alt 3. Apr 2009, 13:53
so, hab oben mal eine Neuere Verion angehängt

einige kleine Änderungen gab's
und vorallem kann man jetzt Nodes einfacher aufrufen/erstellen/prüfen/löschen

Delphi-Quellcode:
XML.RootNode.Nodes.Node['node1'].Nodes.Node['node2'].Nodes.Add('node3')
XML.RootNode.Nodes['node1'].Nodes['node2'].AddNode('node3')
geht jetzt auch
XML.RootNode.AddNode('node1\node2\node3') um in <xml><node1><node2> den <node3> zu erstellen
(nur in IndexOf/IndexOfNF funktioniert dieses "absichtlich" nicht)

ich werd das demnächst auch bei den Namesangaben der Attribute umsetzen

Delphi-Quellcode:
XML.RootNode.Nodes.Node['node1'].Nodes.Node['node2'].Attributes['attr']
XML.RootNode.Nodes['node1'].Nodes['node2'].Attributes['attr']
XML.RootNode.Attributes['node1\node2\attr']
für <xml><node1><node2 attr="">

außerdem soll man dann statt bzw. zusammen mit dem Node-Namen auch einen Parameter mit angeben können

z.B. für den 2. Node:
Delphi-Quellcode:
Nodes['node2']
Nodes['>attr=name2']
Nodes['node2>attr=name2']
<node1 attr="name1">
<node2 attr="name2">
<node3 attr="name3">
  Mit Zitat antworten Zitat
Benutzerbild von bigg
bigg
 
#13
  Alt 14. Apr 2009, 11:16
Hallo himitsu,

deine Klasse lässt sich mit Delphi 7 PE nicht nutzen, da einige Klassen sowie Sprachkonstrukte nicht zur Verfügung stehen.

Zum Beispiel folgendes funktioniert nicht:
Class Operator Implicit
Zitat:
[Fehler] himXML.pas(305): 'END' erwartet, aber 'CLASS' gefunden
[Fehler] himXML.pas(305): '=' erwartet, aber Bezeichner 'Implicit' gefunden
[Fehler] himXML.pas(305): '=' erwartet, aber ':' gefunden
[Fehler] himXML.pas(305): '=' erwartet, aber ';' gefunden
[Fehler] himXML.pas(270): Typ 'TXMLFile' ist nicht vollständig definiert
[Fehler] himXML.pas(271): Typ 'TXMLNode' ist nicht vollständig definiert
[Fehler] himXML.pas(272): Typ 'TXMLNodeList' ist nicht vollständig definiert
[Fehler] himXML.pas(273): Typ 'TXMLAttributes' ist nicht vollständig definiert
[Fehler] himXML.pas(306): PROCEDURE oder FUNCTION erwartet
[Fehler] himXML.pas(306): Bezeichner redefiniert: 'Value'
[Fehler] himXML.pas(306): '=' erwartet, aber ')' gefunden
[Fehler] himXML.pas(307): ';' erwartet, aber 'END' gefunden
[Fehler] himXML.pas(312): Unbekannte Anweisung: 'Public'
[Fehler] himXML.pas(313): Unbekannte Anweisung: 'Str'
[Fehler] himXML.pas(315): Undefinierter Bezeichner: 'TXMLTempData'
[Fehler] himXML.pas(315): '=' erwartet, aber ')' gefunden
[Fehler] himXML.pas(315): '(' erwartet, aber ';' gefunden
[Fehler] himXML.pas(316): Konstantenausdruck erwartet
[Fehler] himXML.pas(316): Ausdruck erwartet, aber 'CONST' gefunden
[Fehler] himXML.pas(316): Bezeichner redefiniert: 'Value'
[Fehler] himXML.pas(316): '=' erwartet, aber ')' gefunden
[Fehler] himXML.pas(317): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(320): Deklaration erwartet, aber Bezeichner 'TXMLWriteBuffer' gefunden
[Fehler] himXML.pas(324): Deklaration erwartet, aber Bezeichner 'TXMLAssembleOptions' gefunden
[Fehler] himXML.pas(330): Bezeichner erwartet, aber '(' gefunden
[Fehler] himXML.pas(330): Prozedur kann keinen Ergebnistyp besitzen
[Fehler] himXML.pas(339): Deklaration erwartet, aber Bezeichner 'TXMLCharCheckTyp' gefunden
[Fehler] himXML.pas(355): Unbekannte Anweisung: 'Static'
[Fehler] himXML.pas(356): Unbekannte Anweisung: 'Static'
[Fehler] himXML.pas(357): Unbekannte Anweisung: 'Static'
[Fehler] himXML.pas(358): Unbekannte Anweisung: 'Static'
[Fehler] himXML.pas(359): Unbekannte Anweisung: 'Static'
[Fehler] himXML.pas(360): Unbekannte Anweisung: 'Strict'
[Fehler] himXML.pas(361): Unbekannte Anweisung: '_Owner'
[Fehler] himXML.pas(394): Unbekannte Anweisung: 'Public'
[Fehler] himXML.pas(395): PROCEDURE oder FUNCTION erwartet
[Fehler] himXML.pas(403): Undefinierter Bezeichner: 'Create'
[Fehler] himXML.pas(404): Undefinierter Bezeichner: 'Destroy'
[Fehler] himXML.pas(406): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(421): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(424): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(438): Unbekannte Anweisung: 'Private'
[Fehler] himXML.pas(444): Undefinierter Bezeichner: 'TXMLAssembleOptions'
[Fehler] himXML.pas(446): Undefinierter Bezeichner: 'TXMLCharCheckTyp'
[Fehler] himXML.pas(447): Undefinierter Bezeichner: 'TXMLStringCheckTyp'
[Fehler] himXML.pas(455): Undefinierter Bezeichner: 'TXMLTempData'
[Fehler] himXML.pas(459): Undefinierter Bezeichner: 'TXMLWriteBuffer'
[Fehler] himXML.pas(470): Deklaration erwartet, aber Bezeichner 'TXMLNodeList' gefunden
[Fehler] himXML.pas(484): Unbekannte Anweisung: 'Private'
[Fehler] himXML.pas(486): Unbekannte Anweisung: 'Public'
[Fehler] himXML.pas(487): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(490): Undefinierter Bezeichner: 'Create'
[Fehler] himXML.pas(491): Undefinierter Bezeichner: 'Destroy'
[Fehler] himXML.pas(493): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(517): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(539): Unbekannte Anweisung: 'Private'
[Fehler] himXML.pas(540): Undefinierter Bezeichner: 'TMXLNodePathOptions'
[Fehler] himXML.pas(547): Deklaration erwartet, aber Bezeichner 'TXMLNode' gefunden
[Fehler] himXML.pas(584): Unbekannte Anweisung: 'Private'
[Fehler] himXML.pas(585): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(587): Unbekannte Anweisung: 'Public'
[Fehler] himXML.pas(588): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(592): Undefinierter Bezeichner: 'Create'
[Fehler] himXML.pas(593): Undefinierter Bezeichner: 'Destroy'
[Fehler] himXML.pas(595): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(617): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(619): '=' erwartet, aber ']' gefunden
[Fehler] himXML.pas(619): '(' erwartet, aber Bezeichner 'Read' gefunden
[Fehler] himXML.pas(619): Nicht genügend wirkliche Parameter
[Fehler] himXML.pas(619): Nicht genügend wirkliche Parameter
[Fehler] himXML.pas(620): Deklaration erwartet, aber 'PROPERTY' gefunden
[Fehler] himXML.pas(620): '=' erwartet, aber ']' gefunden
[Fehler] himXML.pas(620): Konstantenausdruck erwartet
Außerdem ist deine himXML-Unit in UTF-8 kodiert (mit BOM), damit kann D7 natürlich nicht umgehen.
  Mit Zitat antworten Zitat
mkinzler

 
Delphi 11 Alexandria
 
#14
  Alt 14. Apr 2009, 11:20
Klassenoperatoren werden unter Win32 auch erst ab D10 (BDS2006/TD(E)) unterstützt
Markus Kinzler
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

 
Delphi 12 Athens
 
#15
  Alt 15. Apr 2009, 09:32
Jupp, die erste Version war (Aufgrund eines Fehlers, den aber keiner erwähnte) nur in Delphi2009 wirklich lauffähig und die aktuelle Version läuft getestet ab Delphi2006 / Turbo Delphi.

Wenn das Projekt soweit fertig ist und läuft, wird es vermutlich eine abgespeckte und in einigen Properties leicht geänderte(umbenannte) Version geben, welche auch in früheren Versionen lauffähig sein wird. (bis D7 runter kann ich dabei dann selber testen)

Zitat:
Außerdem ist deine himXML-Unit in UTF-8 kodiert (mit BOM), damit kann D7 natürlich nicht umgehen.
Ups, hatte ich zwar bemekrt, daß da Delphi mal, wegen verschiedener Zeichen, umgeschaltet hatte und ich wollte das auch ändern ... hab ich dann wohl total vergessen.
  Mit Zitat antworten Zitat
Muetze1
 
#16
  Alt 15. Apr 2009, 10:15
Wie handhabst du bei Node1/Node2/Node3 eigentlich die Auswahl von gleichnamigen Knoten auf einer Ebene? Also wenn ich mit der zuvor genannten Angabe aber auf der 2. Ebene den 6. Knoten mit dem Namen Node2 haben will?
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

 
Delphi 12 Athens
 
#17
  Alt 15. Apr 2009, 11:09
Bei gleichnamigen Nodes wird der zuerst Gefundene zurückgegeben.

Hmmmm, also daran hatte ich noch garnicht gedacht, aber wenn ich mir das so überleg ... [ und ] sind in Nodenames eh nicht erlaubt, dann könnte ich da ja noch einen Index anzubieten?
Zitat:
Node1/Node2[5]/Node3


Und bezüglich des Problems bei mehrere gleichnamiger Nodes hab auch schon 'ne Weile geplant mal eine Funktion zu erstellen, welche dann ein Array mit all den gefundenen Nodes zurück gibt.
Also daß statt sowas Nodes['Nodename'] dann weißnochnichtwieichdasnenn['Nodename'] nicht den ersten Node liefert, sondern eben ein Array of Node.
  Mit Zitat antworten Zitat
Muetze1
 
#18
  Alt 15. Apr 2009, 11:35
... bzw. eine NodeList. Genauso habe ich es in meiner auch gelöst gehabt - also eine NodeList und die [] in den Pfadangaben für die Indizierung gleichnamiger.
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

 
Delphi 12 Athens
 
#19
  Alt 15. Apr 2009, 11:41
Eine ganze NodeList wollte ich dafür nicht gleich verwenden, denn ansonstenmüßte ich darin entweder das Erstellen/Ändern von Nodes unterbinden oder irgendwie synchronisieren und ein einfaches Array hätte eigentlich doch auch gereicht.
Das Ganze wollte ich ja auch noch "möglichst" Schlank halten :angel
und Aufgrund der Speicherverwaltung nur mit Objekten ist es garnicht so leicht möglich extra eine NodeList dafür zu verwenden, denn wie soll diese wieder freigegeben werden?

Ansonsten zeigen manche Problemchen auch, daß Einiges mit Interfaces wohl leichter zu lösen wäre.
  Mit Zitat antworten Zitat
Muetze1
 
#20
  Alt 15. Apr 2009, 13:41
Zitat von himitsu:
Eine ganze NodeList wollte ich dafür nicht gleich verwenden, denn ansonstenmüßte ich darin entweder das Erstellen/Ändern von Nodes unterbinden oder irgendwie synchronisieren und ein einfaches Array hätte eigentlich doch auch gereicht. :roll:
Hö? Das ist doch sogar sehr einfach. Du reichst in der Liste einfach die Node Instanzen raus, die du findest. Wenn irgendwer diese ändert: Soll er doch - ist doch vollkommen egal, da es die gleiche Instanz ist. Diese Änderungen sind in deiner Struktur drin, da es die gleichen Instanzen sind (um es nochmals zu erwähnen ^^).

Und wenn einer einen Knoten aus der Liste löscht oder freigibt, dann kann sich dieser Knoten doch selbst in seinem Überknoten entfernen. Gleiches hatte ich in meiner Lib auch gemacht - funktioniert einwandfrei. Dadurch war das Knoten löschen recht einfach: einfach freigeben mit .Free und der Knoten samt seinen Unterknoten wird ordentlich freigegeben und die Struktur bei dir im Speicher ist aktuell. Was will man mehr? Das ist doch gerade das schöne an OOP und den Instanzen.

Zitat von himitsu:
Das Ganze wollte ich ja auch noch "möglichst" Schlank halten :angel
Jo, ein Extract()-Aufruf in der Knotenliste des Vaterknoten im Destruktor der Knoten.

Zitat von himitsu:
und Aufgrund der Speicherverwaltung nur mit Objekten ist es garnicht so leicht möglich extra eine NodeList dafür zu verwenden, denn wie soll diese wieder freigegeben werden?
Oh und gerade deshalb ja.

Objekte sollten immer auf der gleichen Ebene freigegeben werden wo sie auch alloziiert werden. Also kannst du schonmal keine Rückgabeliste erzeugen, also bekommst du eine vom Nutzer übergeben. Der legt sie an und gibt sie auf der gleichen Ebene wieder frei.

Schau dir doch einfach meine Lib an in Sachen Objektverwaltung - zur Not kopier es dir sogar. Die Lib ist (öffentlich) tot und du kannst dich frei bedienen.
  Mit Zitat antworten Zitat
Themen-Optionen Thema durchsuchen
Thema durchsuchen:

Erweiterte Suche
Ansicht

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:

(?)

LinkBack to this Thread

Erstellt von For Type Datum
xml - MSXML alternative - Stack Overflow This thread Refback 28. Jun 2011 15:34

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 04:12 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