Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Delphi speichersparendes StringList.SaveToFile (https://www.delphipraxis.net/140035-speichersparendes-stringlist-savetofile.html)

himitsu 10. Sep 2009 21:04


speichersparendes StringList.SaveToFile
 
Ich hatte grad den Fall, daß eine StringListe mit nur 10928163 Zeilen (~1,8 GB Arbeitsspeicher)
sich nicht speichern ließ. :freak:

Der Grund war, daß SaveToFile (Strings/TStringList) erstmal alles zu einem einzigen String zusammensetzen und in neueren Delphis danach auch noch umkodieren will.

Da war es ja klar, daß dieses nicht nochmal in die verbleibenden ~200 MB reinpaßt. :angel2:

Also wenn es mal wer braucht, gibt es hier eine speichersparende Alternative:
Delphi-Quellcode:
// nach dem Vorbild von Delphi 7
procedure SaveToStreamEx(Strings: TStrings; Stream: TStream);
var
  S: String;
  i: Integer;
begin
  try
    S := Strings.Text;
  except
    for i := 0 to Strings.Count - 1 do begin
      S := Strings[i] + sLineBreak;
      Stream.WriteBuffer(Pointer(S)^, Length(S));
    end;
    Exit;
  end;
  Stream.WriteBuffer(Pointer(S)^, Length(S));
end;

// nach dem Vorbild von Delphi 2009
procedure SaveToStreamEx(Strings: TStrings; Stream: TStream; Encoding: TEncoding = nil);
var
  Buffer, Preamble: TBytes;
  i: Integer;
begin
  if Encoding = nil then
    Encoding := TEncoding.Default;
  Preamble := Encoding.GetPreamble;
  if Length(Preamble) > 0 then
    Stream.WriteBuffer(Preamble[0], Length(Preamble));
  try
    Buffer := Encoding.GetBytes(Strings.Text);
  except
    for i := 0 to Strings.Count - 1 do begin
      Buffer := Encoding.GetBytes(Strings[i] + sLineBreak);
      Stream.WriteBuffer(Buffer[0], Length(Buffer));
    end;
    Exit;
  end;
  Stream.WriteBuffer(Buffer[0], Length(Buffer));
end;
Es wird erstmal der "alte" Weg versucht und bei einer Exception (OutOfMemory) wird der andere Weg versucht.

Hier das Ganze dann nochmals nur über den speichersparenden Weg.
Delphi-Quellcode:
procedure SaveToStreamLowMem(Strings: TStrings; Stream: TStream);
var
  S: String;
  i: Integer;
begin
  for i := 0 to Strings.Count - 1 do begin
    S := Strings[i] + sLineBreak;
    Stream.WriteBuffer(Pointer(S)^, Length(S));
  end;
end;

procedure SaveToStreamLowMem(Strings: TStrings; Stream: TStream; Encoding: TEncoding = nil);
var
  Buffer, Preamble: TBytes;
  i: Integer;
begin
  if Encoding = nil then
    Encoding := TEncoding.Default;
  Preamble := Encoding.GetPreamble;
  if Length(Preamble) > 0 then
    Stream.WriteBuffer(Preamble[0], Length(Preamble));
  for i := 0 to Strings.Count - 1 do begin
    Buffer := Encoding.GetBytes(Strings[i] + sLineBreak);
    Stream.WriteBuffer(Buffer[0], Length(Buffer));
  end;
end;

Manchmal ist es ja effektiver den alten Weg zu nehmen, wenn es denn möglich ist.
z.B. bei TMemo, wo der Text schon im Ganzen in der Komponente daliegt und der zeilenweise Zugriff langsamer wäre.

[edit]
die Schleifenfariable in den 2009er-Versionen vergessen zu defineren :oops:


Alle Zeitangaben in WEZ +1. Es ist jetzt 00:25 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