Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Sonstige Fragen zu Delphi (https://www.delphipraxis.net/19-sonstige-fragen-zu-delphi/)
-   -   Dateien auf Memorystick schreiben extrem langsam (https://www.delphipraxis.net/164766-dateien-auf-memorystick-schreiben-extrem-langsam.html)

iphi 29. Nov 2011 09:46

Dateien auf Memorystick schreiben extrem langsam
 
Hallo,

mein Programm produziert fortlaufend Daten in Form von Textdateien, die fortlaufend auf die Festplatte geschrieben werden, und zwar im Prinzip so:

Delphi-Quellcode:
var F: TextFile;
...
AssignFile(F,NextName);
Rewrite(F);
For i:=1 to n do
  begin
    datenzeile:=...
    writeln(F, datenzeile);
    ...
  end;
CloseFile(F);
...
Das funktioniert auch. Auf Festplatte benötigt das Schreiben eines Datenfiles akzeptable 20ms.
Dassselbe auf USB-Stick benötigt ca. 2 Sekunden, also einen Faktor von 100 länger!

Wenn ich Dateien auf Festplatte bzw. den USB Stick kopiere, dann ist der Geschwindigkeitsunterschied deutlich geringer (Übertragungsrate auf Festplatte nur um Faktor 10 schneller als auf USB-Stick).

Weiß jemand, woher die Diskrepanz Faktor 10 ggü Faktor 100 kommen kann?
Wie kann ich effizienter mit meinem Programm auf den USB-Stick schreiben?

DeddyH 29. Nov 2011 10:00

AW: Dateien auf Memorystick schreiben extrem langsam
 
Geht es mit einer Stringliste oder einem Stream schneller?

himitsu 29. Nov 2011 11:21

AW: Dateien auf Memorystick schreiben extrem langsam
 
Delphi-Referenz durchsuchenTextFile hat eine "kranke" Einstellung der Puffergröße von (maxinmal) 128 Byte.

Wenn der der USB-Stick auf "schnelles Entfernen" eingestellt ist, dann wird alles sofort gespeichert.

Zugriffe auf Datenträger sind immer nur in ganzen Sektoren möglich, was 512 Byte entspricht.
Bei einer Sektorgröße von 512 Byte wird nun also jeder Cluster mindestens 4 Mal auf den Stick geschrieben.

Eventuell obtimiert das Windows aber aug ganze Cluster (mehrere zusammengefaßte Sektoren).
Bei einer Clustergröße von z.B. 8 KB wird nun also vermutlich jeder Cluster gleich 32 Mal auf den Stick geschrieben.
Konnte aber auch sein, daß Windows doch nicht ganze Cluster, sondern nur in ganzen Sektoren schreibt, aber vermutlich nicht, da die Cacheverwaltung bestimmt mit ganzen Clustern arbeitet.

Bei einer vollen Cache, bzw. bei zuwenig RAM muß auch noch oftmals der aktuelle Cluster/Sektor ausgelesen werden, bevor er gespeichert wird, da er ja nur Teilweise beschrieben wird.

Bei einem stückchenweisen befüllen der Datei wird, ebenfalls beim "schnellen Entfernen", wird zusätzlich auch für jedes geschriebenes Stückchen die Dateiinfo (Dateigröße) und oftmals auch noch die MemoryMap (Belegungsliste aller Cluster) erneut geschrieben, was dann für eine Änderung von jedem Cluster Zugriffe auf mindestens 3 Cluster bedeutet.

Ein großer/gemeinsamer Schreibzugriff kann also nur verbessern.

Lösungen:
- einen größeren Buffer einstellen
- eine StringList/Stream verwenden, welche nur einmal am Ende alles zusammen rüberschicken

gammatester 29. Nov 2011 12:01

AW: Dateien auf Memorystick schreiben extrem langsam
 
Zitat:

Zitat von himitsu (Beitrag 1138396)
Delphi-Referenz durchsuchenTextFile hat eine "kranke" Einstellung der Puffergröße von (maxinmal) 128 Byte.

Nicht maximal, sondern default. Für größere Puffer gibt's
Delphi-Quellcode:
procedure SetTextBuf(var F: Text; var Buf; Size: Integer);

himitsu 29. Nov 2011 13:44

AW: Dateien auf Memorystick schreiben extrem langsam
 
Ja, der Buffer hat 128 Byte (per Default), aber wenn ein Write-Befehl nicht mehr in diesen Buffer paßt, dann wird der Buffer vorzeitig geleert und der WriteBefehl eventuell sogar direkt geschrieben, ohne durch den Buffer zu gehn.

= also maximal, da man selten die 128 genau treffen wird (außer man schreibt byteweise)

himitsu 29. Nov 2011 13:45

AW: Dateien auf Memorystick schreiben extrem langsam
 
Zitat:

Zitat von gammatester (Beitrag 1138402)
Zitat:

Zitat von himitsu (Beitrag 1138396)
Delphi-Referenz durchsuchenTextFile hat eine "kranke" Einstellung der Puffergröße von (maxinmal) 128 Byte.

Nicht maximal, sondern default. Für größere Puffer gibt's
Delphi-Quellcode:
procedure SetTextBuf(var F: Text; var Buf; Size: Integer);

Dieses Verhalten tritt auch mit einem größeren Buffer ein ... es kommt also immer dazu, daß irgendwo nur Teile von Sektoren geschrieben werden.
Über die größe des Buffers kann man nur die Häufigkeit regeln.


zum Default
Zitat:

- einen größeren Buffer einstellen


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