![]() |
AW: Mehrdimensionales, dynamisches Array speichern
naja, ich machs mal etwas Pseudomäßig...
Code:
=> als CSV, getrennt durch TAB und Zeilenumbruch
Array
( Array ('e1', 'e2', 'e3'), Array ('f1', 'f2', 'f3') )
Code:
Dies wäre die einfachste Variante.
e1 e2 e3
f1 f2 f3
Delphi-Quellcode:
filecontent := '';
for i := 0 to length (Array) - 1 do begin for j := 0 to length (Array[i]) - 1 do begin filecontent := filecontent + Array[i][j]; if (j + 1 < length (Array[i])) then begin filecontent := filecontent + #9; end; end; if (i + 1 < length (Array)) then begin filecontent := filecontent + #13#10; end; end; //WriteToFile... (filename, filecontent); |
AW: Mehrdimensionales, dynamisches Array speichern
Zeilenweise einlesen (ReadLn oder über eine TStringList) und dann könnte man erstmal die Zeilen zerlegen (manuell oder z.B. über
![]() |
AW: Mehrdimensionales, dynamisches Array speichern
Okay, danke soweit =)
EDIT: Wo kann ich anchlesen wie das mit der Stringlist funktioniert (Strings ausplitten), ich hab hier im Forum nur andere Varianten gefunden. Wenn die Information in der ersten Zeile 1;1;A;0 beträgt, wie komme ich dann an das 'A' dran? EDIT: Ich hab es (wahrscheinlich falsch) mit dem Explode von alzaimar probiert und meine 4GB Ram voll ausgelastet...
Delphi-Quellcode:
procedure laden;
var Datei: TextFile; Zeile: string; i, n : integer; StringList1, sSubStrings : TStringList; MyDivider: TStringDivider; MyIterator: TStringDivideIterator; begin StringList1 := TStringList.Create; sSubStrings := TStringList.Create; try StringList1.LoadFromFile(ExtractFilePath(Application.ExeName)+'text.CSV'); for i := 1 to AnzahlZeilen do for n := 1 to AnzahlSpalten do begin MyDivider := TStringDivider.Create; MyDivider.Pattern := ';'; MyDivider.Explode(StringList1.Strings[i], sSubStrings); MyIterator := TStringDivideIterator.Create; MyIterator.Pattern := ';'; MyIterator.Text := StringList1.Strings[i]; While MyIterator.MoveNext do sSubStrings.Add(MyIterator.CurrentStr); end; finally StringList1.Free; sSubStrings.Free; MyDivider.Free; MyIterator.Free; end; end; |
AW: Mehrdimensionales, dynamisches Array speichern
Das ist ganz schön krass was du da zusammen würfelst. Unabhängig davon ob deine Logik funktioniert erzeugst du mit dem Beispiel AnzahlZeilen x AnzahlSpalten Speicherlecks!
Denn du erstellst innerhalb der Schleife jedesmal ein Objekt. Mit nächstem Durchlauf überschreibst du gnadenlos die Referenz. Und nach der Schleife gibst du nur ein Objekt frei? Und das ganze sogar zweimal... Setze mal mit dem Anwendungsstart (DPR)
Delphi-Quellcode:
und lass deine Funktion durchlaufen und beende dein Programm anschließend (wenn dies überhaupt ohne Zugriffsfehler möglich ist).
ReportMemoryLeaksOnShutdown := true
Wenn du gar nicht begreifst was ich meine dann starte mal deine Funktion 20 mal und schau dir im Taskmanager die Speicherauslastung an. Und wenn du jetzt sagst, was solls... Speicheranforderungen sind Systemaufrufe und die kosten viel viel Zeit! Und das in einer Schleife? Und wieso erstellst du die Objekte oberhalb vom try-finally-Block? Und woher willst du im finally-Block wissen, dass alle Objekte auch wirklich existieren? Was ist deine Zieldatei nicht existiert oder das ganze inmitten der Schleifen abbricht? Worauf zeigt dann MyIterator? Und worauf die Methode Free ()? Oder anderen Sachen? Vielleicht geht alles noch beim Lesen der Datei irgendwie gut. Jedoch wenn du auf deinem System irgendwann nur noch halbe Dateien findest solltest du dich nicht wundern. Du solltest dir eine Funktion explode ('delimiter', 'content') schreiben die möglichst gar keine Objekte verwendet und nichts anderes macht als einen String in Teile zu zerlegen. |
AW: Mehrdimensionales, dynamisches Array speichern
@ASM
danke ... aber hast Du mal meine und Deine Varianten Debugt? IMHO sind meine die korrekten. |
AW: Mehrdimensionales, dynamisches Array speichern
Zitat:
Du kannst nicht die Fehlerfreiheit einer Codierung dadurch korrekt prüfen und angeblich nachweisen, wenn Du - wie Du es gemacht hast - die relativ komplizierte Struktur eines 2-dimensionalen Arrays an gerade mal einer einzigen Elementposition des Arrays testest. Das ist nicht eine Frage rein theoretischer Interpretation, sondern vor allem eine der praktischen Prüfung. Auf den Fehler in Deinem Code bin ich nämlich gerade dadurch aufmerksam geworden, nachdem ich das komplette 2D-Array mit unterschiedlichen Werten gefüllt hatte und dann folgendermaßen vorgegangen bin: zum einen nach programminternem Austausch per Memorystream zwischen zwei formal gleichen Arrayvariablen (A und B), zum anderen nach Abspeicherung des ersten Arrays (A) per Filestream in einem File und anschließenden Einlesen dieses Files per Filestream in das zweite Array (B), wobei sich ergänzend mit einem Hexeditor die im File gespeicherte Information exakt nachprüfen ließ. Ergebnis: Mit Deinem originalen Code wurde in beiden Verfahren nur Garbage sichtbar, und außerdem führte es zufallsbedingt an unregelmäßigen Arraypositionen zu Crash wegen Access violation. Wie Du also zu Deiner jetzigen Verteidigung und Behauptung kommen kannst, ist mir rätselhaft. Vielleicht solltest Du es einmal kontrollieren mit meiner folgenden Erweiterung Deines Beispielcodes zur Anwendung:
Code:
Was passiert wohl, wenn Du das Ergebnis der entsprechende Anwendung unter Verwendung Deiner Originalunit vergleichst mit der von mir korrigierten Version der Unit ?
procedure TForm1.Button2Click(Sender: TObject);
var a,b:T2DStringArray; i:Integer; ms:TMemoryStream; begin ms:=TMemoryStream.Create; try SetLength(a,100,0); for I := Low(a) to High(a) do begin // hier ist im Original (SetLength(a[i],i) schon einmal ein Fehler: // bei i=0 wäre das Array a[0,0] an dieser Stelle sonst nur 1-dimensional!, // der Versuch einer Belegung führt unweigerlich zum Crash // also muss entweder die Länge der zweiten Dimension auf mindestens 1 // gesetzt werden oder es müsste eine Sicherheitsabfrage auf das Element A[i,0] // erfolgen SetLength(a[i],i+1); a[i][random(i)]:=char(32+1); end; a[19,5] := 'Haus'; a[59,2] := 'Dies ist ein Test'; Save2DArray(a,ms); // hierdurch wird unmissverständlich sichtbar, // wie verheerend der originale Code sich auswirkt! // Da das Array A ja ab hier nicht mehr benötigt wird, // darf seine Eliminierung in der folgenden Anzeige der Elemente des Array B // keinerlei Folgen haben; ist das so bei Deinem Code ? setlength(a,0,0); ms.Position := 0; Load2DArray(b,ms); Showmessage(format('%s'#13#10'%s',[b[19,5], b[59,2]])); finally ms.Free; end; end; Ach ja, und im übrigen stammt der von Dir gepostete Code in seinem Kern eigentlich nicht von Dir (bzw. von einem Thomas Wassermann ?), sondern von einem P.Below aus dem Jahre 2003, der den Code unter dem Titel "Save and load a 2-dimensional dynamic array" in Torry's Delphi Pages veröffentlicht hatte ( ![]() |
AW: Mehrdimensionales, dynamisches Array speichern
Ja, ich war wohl recht müde ....
[delphi] writer.Write(A[i, x][1], SizeOf(Char) * Length(A[i,x])); //bzw. reader.Read(A[i, x][1],SizeOf(Char) * slen); [delphi] ist der richtige Code. |
AW: Mehrdimensionales, dynamisches Array speichern
Was gegen, wenn ich den hier mal ein Anstoß einwerfe???
Delphi-Quellcode:
sollte sich eigendlich von selbst erklären. must nur die variablen an deines anpassen.
procedure TRegel.SetYear(AValue: TYear);
var d, m: integer; s: string; begin FYear := AValue; FFilename := Format('%s%d.%s', [FPath, FYear.Yaer, FExt+FYear.Benutzer.Name+' '+FYear.Benutzer.Lastname]); if FileExists(FFilename) then begin // Daten laden with TIniFile.Create(FFilename) do try for m := 1 to 12 do begin for d := 1 to 31 do begin s := Format('%.2d-%.2d', [m, d]); Items[d, m].HasRegel := ReadString(s, 'HasRegel', 'false'); end; end; finally Free; end; end; end; zur not bitte via PN nachfragen. würde denne in tewa so aussehen.
Code:
zugegeben. ist ein Programmausschnitt aus meinem Aktuellen Projekt "Mestruationskalender"
[01-05]
HasRegel=1 Stärke=Normal Schmerz=Keine [01-06] HasRegel=1 Stärke=Normal Schmerz=Keine [01-07] HasRegel=1 Stärke=Normal Schmerz=Keine [01-08] HasRegel=1 Stärke=Leicht Schmerz=Keine |
AW: Mehrdimensionales, dynamisches Array speichern
Nein, TIniFile ist nicht für viele Daten gedacht, welche sich auch noch oft ändern.
(Begründung siehe ![]() PS: "HasRegel" schreibt man mit Doppel-S oder mit T. |
AW: Mehrdimensionales, dynamisches Array speichern
@himi:
Zitat:
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 02:41 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