![]() |
ReadFile und WriteFile
Hallo,
ich habe ein Problem mit dem Befehl WriteFile. Ich benutze Delphi 6. Wenn ich die folgenden Befehle ausführen lasse, bekomme ich immer nur eine 0-Byte-Datei:
Delphi-Quellcode:
Die Lese-/Schreibrechte sind nicht das Problem. Die Datei wird angelegt/überschrieben. Es muss also an WriteFile liegen.
Var
NR,SB : Cardinal; Buffer : Array [1..5120] Of Byte; FHandle : Cardinal; Begin SB:=SizeOf(Buffer); FHandle:=CreateFile(PChar('C:\Test.$$$'), GENERIC_READ or GENERIC_WRITE, 0, nil, CREATE_ALWAYS, FILE_FLAG_NO_BUFFERING or FILE_FLAG_WRITE_THROUGH, 0); If FHandle <> INVALID_HANDLE_VALUE Then Begin WriteFile(FHandle,Buffer,SB,NR,nil); FileClose(FHandle); End; End; 1. Was mache ich falsch? 2. Wo gibt es eine schöne (und ggf. deutsche) Beschreibung für CreateFile, WriteFile, ReadFile etc.? |
Re: ReadFile und WriteFile
Hallo,
ich würde den Rückgabewert von WriteFile() beobachten. So richtig etwas falsches habe ich nicht sehen können, darum habe ich den Code mal mit D7 übersetzt - und er tut was er soll.
Delphi-Quellcode:
Grüße vom marabu
Begin
SB := SizeOf(Buffer); FHandle := CreateFile( 'C:\daten\DP\Test.$$$', GENERIC_READ or GENERIC_WRITE, 0, nil, CREATE_ALWAYS, FILE_FLAG_NO_BUFFERING or FILE_FLAG_WRITE_THROUGH, 0 ); if FHandle <> INVALID_HANDLE_VALUE then begin if WriteFile(FHandle, Buffer, SB, NR, nil) then ShowMessage(IntToStr(NR)) else ShowMessage(SysErrorMessage(GetLastError)); FileClose(FHandle); end; end; |
Re: ReadFile und WriteFile
Es ist nur inkonsequent, am Ende FileClose aufzurufen. Zu CreateFile gehört CloseHandle. FileClose macht zwar intern auch nichts anderes, aber FileClose gehört eigentlich zu FileOpen.
|
Re: ReadFile und WriteFile
Moin Schwedenbitter,
da kann ich mich Marabu nur anschliessen. Funktioniert problemlos. Bei mir auch mit D7 getestet, unter XP Pro SP 2, und voll gepatcht. Mit welchem Betriebssystem arbeitest Du? |
Re: ReadFile und WriteFile
@ Alle
Es stimmt, der Code funktioniert auch bei mir. Erstmal danke für den Tipp mit SysErrorMessage(GetLastError). Das kannte ich nicht. Allerdings ist mein Code von oben nur ein Ausschnitt. Wenn ich den nehme, mit Delphi ein neues Programm aufmache, den Teil reinkopiere und ausführen lasse, dann klappt es. Der Fehler muss also am Rest des Codes liegen. Deshalb hier mal alles:
Delphi-Quellcode:
Wie gesagt: Das Anlegen klappt, nur die Datei ist leer.
Procedure TForm1.Start1Click(Sender: TObject);
Var I : Word; NR,SB : Cardinal; Buffer : Array [1..5120] Of Byte; FHandle : Cardinal; E1,E2,CB : Boolean; Begin Ende:=False; SB:=SizeOf(Buffer); {Tastenstatus merken und deaktivieren} E1:=Edit1.Enabled; Edit1.Enabled:=False; E2:=Edit2.Enabled; Edit2.Enabled:=False; CB:=CB1.Enabled; CB1.Enabled:=False; Start1.Visible:=False; Stop1.Visible:=True; {Puffer mit Zufallszahlen füllen} Randomize; FillChar(Buffer,SB,0); For I:=1 To SB Do Begin Buffer[I]:=Random(256); End; {Datei anlegen} FHandle:=CreateFile(PChar(Edit1.Text), GENERIC_READ or GENERIC_WRITE, 0, nil, CREATE_ALWAYS, FILE_FLAG_NO_BUFFERING or FILE_FLAG_WRITE_THROUGH, 0); { Jetzt schreiben} If FHandle<>INVALID_HANDLE_VALUE Then Begin If CB1.Checked Then Begin Repeat WriteFile(FHandle,Buffer,SB,NR,nil); Application.ProcessMessages; Until (NR<SB) Or Ende; End Else Begin If WriteFile(FHandle,Buffer,SB,NR,nil) Then ShowMessage(IntToStr(NR)) Else ShowMessage('Mist: '+SysErrorMessage(GetLastError)); End; {Datei schließen} //FileClose(FHandle); End; {Tastenstatus wiederherstellen} Edit1.Enabled:=E1; Edit2.Enabled:=E2; CB1.Enabled:=CB; Start1.Visible:=True; Stop1.Visible:=False; End; Edit1 ist TEdit und fragt den Dateinamen ab. Edit2 ist TEdit und fragt die Größe ab -> noch nicht implementiert. CB1 ist TCheckBox und gibt an, ob der Datenträger voll (NR<SB) geschrieben werden soll: Beide Varianten produzieren nur eine 0-Byte-Datei. Existiert die Datei nicht und wird neu angelegt, bringt mir SysErrorMessage(GetLastError) die Meldung Falscher Parameter. Was mir das sagen soll, weiß ich nicht. Wenn die Datei nur überschrieben wird, kommt überhaupt keine Meldung <--- Das verstehe ich nun überhaupt nicht. Nach meinem begrenzten Verständnis müsste doch immer eine Meldung kommen: Entweder mit Fehler oder mit den geschriebenen Bytes. Das Programm stürzt auch nicht ab, sondern tut weiter so als wäre nichts gewesen... XP SP3 (32bit) mit allen Patches nach SP3. Dankbar für weitere Ideen und Gute Nacht Alex |
Re: ReadFile und WriteFile
Moin Alex,
wie sieht denn bei D6 die Deklaration von WriteFile aus? In D7 sieht es so aus:
Delphi-Quellcode:
function WriteFile(hFile: THandle; const Buffer; nNumberOfBytesToWrite: DWORD;
var lpNumberOfBytesWritten: DWORD; lpOverlapped: POverlapped): BOOL; stdcall; |
Re: ReadFile und WriteFile
Moin,
die API Funktion WriteFile hat sich wohl nie geändert und D6 übersetzt den Code ohne Fehler, also wird schon alles in Ordnung sein. Durch das Verarbeiten von Nachrichten innerhalb der Schleife soll wohl dem Benutzer des Programms eine Abbruch-Möglichkeit gegeben werden. Das dürfte aber bei vernünftiger Handhabung keinen Einfluß auf die Dateigröße nehmen. Wenn FileClose() allerdings immer auskommentiert war, dann ist eine leer Datei ein durchaus zu erwartendes Ergebnis. Beim Grübeln konnte ich es mir nicht verkneifen den Code etwas umzugestalten.
Delphi-Quellcode:
Freundliche Grüße
const
BUFSIZE = 5120; var // ... buffer: array [1..BUFSIZE] of Bytes; writeOnce, writeFailed, writeCanceled: Boolean; bytesWritten: Cardinal; begin // ... writeCanceled := False; writeOnce := not CB1.Enabled; if FHandle <> INVALID_HANDLE_VALUE then begin repeat writeFailed := not WriteFile(FHandle, buffer, BUFSIZE, bytesWritten, nil); Application.ProcessMessages; until writeOnce or writeFailed or writeCanceled or (BytesWritten < BUFSIZE); if writeFailed then ShowMessage('Mist: ' + SysErrorMessage(GetLastError)); CloseHandle(FHandle); end; // ... end; |
Re: ReadFile und WriteFile
Zitat:
Dass mit dem korrigieren des Codes ist OK. Ich will ja was lernen. Die Deklaration von WriteFile ist auch bei Delphi 6 schon so. Ich werde es heute Abend wieder probieren. Bin jetzt auf Arbeit und habe hier kein Delphi :-( Schönen Tag wünscht Alex |
Re: ReadFile und WriteFile
Schade:
Ich habe die Änderungen in meinen Quellcode übernommen. Er schreibt die Datei trotzdem nicht. Da der Code doch etwas länger ist, habe ich hier mal die ![]() Auf WriteFile() bin ich nur gekommen, weil man da mit den entsprechenden Optionen versuchen kann, Windows zum Schreiben ohne Cache etc. zu zwingen... Gruß, Alex |
Re: ReadFile und WriteFile
Hi Alex,
Zitat:
Ich habe das dann mal für dich gemacht und dann sieht das so aus:
Delphi-Quellcode:
Ein MegaByte musst du nicht selbst ausrechnen, das macht der Compiler für dich - wenn du willst. Den Buffer solltest du nicht lokal zur Prozedur deklarieren, er könnte den Stack sprengen. Beim Benennen deiner Komponenten und Variablen bist du noch nicht sehr konsistent, lies dazu mal einen
Const
BuffSize = 1 shl 20; // genau 1,0 MB var Buffer: Array [1..BuffSize] Of Byte; Procedure TForm1.Start1Click(Sender: TObject); Var I: Cardinal; FHandle: Cardinal; WriteFailed, WriteCanceled: Boolean; BytesWritten: Cardinal; E1,E2,CB: Boolean; begin // ... { Jetzt schreiben} If FHandle<>INVALID_HANDLE_VALUE Then Begin if CB1.Checked then I := MaxInt else I := StrToInt(Edit2.Text); Repeat WriteFailed:=not WriteFile(FHandle,Buffer,BuffSize,BytesWritten,nil); Dec(I); Application.ProcessMessages; Until WriteCanceled Or WriteFailed Or (I=0) or (BytesWritten<BuffSize); If WriteFailed Then ShowMessage('Mist: ' + SysErrorMessage(GetLastError)); {Datei schließen} CloseHandle(FHandle); End; // ... end; ![]() Vergleichbare Ergebnisse erhältst du mit deiner Programmidee nie. Du kannst zwar den einen oder anderen Cache eliminieren, aber die aktuelle Verteilung von freiem Speicherplatz dürfte von Festplatte zu Festplatte stark abweichen. Trotzdem noch viel Spaß mit deinem Projekt. Freundliche Grüße |
Alle Zeitangaben in WEZ +1. Es ist jetzt 21:45 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