![]() |
Re: [TStringList] Problem mit einer Datei (Lädt nicht)
Guten Morgen,
auch die Methoden SaveToFile() und LoadFromFile() von TStrings arbeiten mit Textdateien - und mit dem Zeilentrenner CR/LF für solche Dateien. Du möchtest beliebige Zeichen (#0..#255) in deiner Datei speichern, wie man auch an deiner Musterdatei sehen kann. Kritisch sind dabei solche Zeichen wie #0 (NULL) #13 (CR) #10 (LF) und gelegentlich #26 (EOF), weil sie bei der Verarbeitung von Text als Steuerzeichen interpretiert werden. Das Schlüsselwort PACKED kannst du mit strukturierten (record) und wiederholten (array) Feldern benutzen um die Ausrichtung einzelner Felder an Doppelwortgrenzen zu verhindern, was zwar das Laufzeitverhalten günstig beeinflusst, aber den Speicherverbrauch stark erhöht. Mir ist eine kompakte Speicherung wichtig, da ich die Matrix als Speicherblock behandeln möchte. Wenn die Dimensionen deiner drei-dimensionalen Matrix nicht feststehen, dann musst du sie mit in die Datei schreiben, vorzugsweise als Header davor:
Delphi-Quellcode:
Freundliche Grüße
const
LEVELS = 4; ROWS = 12; COLS = 12; type TDimArray = array [0..3] of Byte; TMatrix = packed array of array of array of Byte; procedure RedimMatrix(var m: TMatrix; dim: array of Byte); begin SetLength(m, LEVELS, ROWS, COLS); end; procedure SaveMatrix(m: TMatrix; fn: TFileName); var dwDim, dwSize: Cardinal; begin dwDim := (Length(m) shl 8 + Length(m[0])) shl 8 + Length(m[0, 0]); dwSize := Length(m) * Length(m[0]) * Length(m[0, 0]); with TFileStream.Create(fn, fmCreate) do try Write(dwDim, SizeOf(dwDim)); Write(m[0], dwSize); // korrigiert finally Free; end; end; procedure LoadMatrix(var m: TMatrix; fn: TFileName); var dwDim, dwSize: Cardinal; dim: TDimArray absolute dwDim; begin with TFileStream.Create(fn, fmOpenRead or fmShareDenyWrite) do try Read(dwDim, SizeOf(dwDim)); // SetLength(m, Hi(dwDim), Lo(dwDim) shr 8, dwDim and $FF); // fehlerhaft SetLength(m, dim[2], dim[1], dim[0]); dwSize := Length(m) * Length(m[0]) * Length(m[0, 0]); Read(m[0], dwSize); // korrigiert finally Free; end; end; { procedure TestMatrix; const FN = 'c:\daten\dp\map.f2m'; var m: TMatrix; iLevel, iRow, iCol: Integer; begin RedimMatrix(m, [LEVELS, ROWS, COLS]); for iLevel := Low(m) to High(m) do for iRow := Low(m[0]) to High(m[0]) do for iCol := Low(m[0, 0]) to High(m[0, 0]) do m[iLevel, iRow, iCol] := 48 + iLevel * 4 + iRow * iCol mod 10; SaveMatrix(m, fn); FillChar(m, SizeOf(m), 0); LoadMatrix(m, fn); end; } procedure TestMatrix; const FN = 'c:\daten\dp\map.f2m'; var m: TMatrix; iLevel, iRow, iCol: Integer; dwSize: Cardinal; okay: Boolean; begin RedimMatrix(m, [LEVELS, ROWS, COLS]); dwSize := LEVELS * ROWS * COLS; for iLevel := Low(m) to High(m) do for iRow := Low(m[0]) to High(m[0]) do for iCol := Low(m[0, 0]) to High(m[0, 0]) do m[iLevel, iRow, iCol] := 48 + iLevel * 4 + iRow * iCol mod 10; SaveMatrix(m, fn); FillChar(m[0], dwSize, 0); LoadMatrix(m, fn); okay := True; for iLevel := Low(m) to High(m) do for iRow := Low(m[0]) to High(m[0]) do for iCol := Low(m[0, 0]) to High(m[0, 0]) do if m[iLevel, iRow, iCol] <> 48 + iLevel * 4 + iRow * iCol mod 10 then okay := False; WriteLn(IfThen(okay, 'fine', 'rats')); end; |
Re: [TStringList] Problem mit einer Datei (Lädt nicht)
Zitat:
Zitat:
Zitat:
Also müsste ich nichts im Header schreiben ;) Obwohl ich dann das Problem bei #10 und #13 hätte, da die ja ein Zeilenumbruch verursachen... Also doch Header :D Zitat:
Zitat:
1. Was ist shl? 2. Soll dwDim der Header sein? 3. Was ist "SizeOf"? Soetwas wie "Length"? Zitat:
1. dwDim hat ja noch keine Größe, aber warum funktioniert dann SizeOf? 2. Hi = High? Lo = Low? Oder was für Funktionen sind das? Zitat:
|
Re: [TStringList] Problem mit einer Datei (Lädt nicht)
Hallo Fabian,
für die gängigen Mikro-Prozessoren sind Zugriffe auf Speicheradressen vorteilhaft, die ein Vielfaches der Registerbreite darstellen. Mit PACKED wird die vom Delphi-Compiler aus Performanzgründen (eventuell) vorgenommene Spreizung der einzelnen Felder eines strukturierten Datentyps verhindert. Wenn die Dimensionen feststehen, dann musst du nicht mit einem dynamischen Array arbeiten. Den Header mit den Dimensionen würde ich wohl auch dann schreiben, aber die Redimensionierung in der Routine LoadMatrix() würde ich durch eine Dimensionsprüfung ersetzen. Probleme mit #13 und #10 hast du immer - wenn du an der Textdatei festhalten willst. "n SHL 8" ist das gleiche wie "n * 256" - oder auch "n * 2 hoch 8". Die Online-Hilfe von Delphi beschreibt die Operatoren SHL und SHR ganz gut. dwDIM ist tatsächlich der Header - ein Vier-Byte-Wert, bei dem wegen deiner drei Dimensionen ein Byte nicht genutzt wird. SizeOf(m) gibt den von m belegten Nettospeicher in Bytes an. Length(m) würde dir Größe der ersten Dimension des Arrays liefern. SizeOf(dwSize) funktioniert also immer und liefert 4 - wie auch SizeOf(Integer). Hi() und Lo() sind in meinem Code fehlerhaft angewendet, ich habe die Zeile geändert. Grüße |
Re: [TStringList] Problem mit einer Datei (Lädt nicht)
Hallo marabu,
hast du übersehen, daß bei dynamischen Arrays nur ein Zeiger auf die Verwaltungsinformationen in der Referenzvariablen abgelegt wird? SizeOf(m) liefert also tatsächlich nur den Wert 4 - die Größe eines Pointers. Beim Aufruf von FileStream.Read, TFileStream.Write und FillChar muß also die Adresse des ersten Feldelements übergeben werden. Bei FillChar ist zusätzlich der zweite Parameter anzupassen. Gruß Hawkeye |
Re: [TStringList] Problem mit einer Datei (Lädt nicht)
Hallo Hawkeye,
zwar habe ich nach der Umstellung meines Codes von statischem auf dynamisches Array die Routine SaveMatrix() angepasst, aber offensichtlich einige Stellen übersehen. Freundliche Grüße PS: Habe (hoffentlich) alle notwendigen Änderungen nachgetragen. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 14:12 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