Frage zu ein Beispielprogramm aus einen Buch
-
Mein Englisch ist nicht so gut, deshalb an die die den Standard besser verstehen.
Furble Wurble schrieb:
Nice Catch!
Ja, nach w+b ist das File leer.
Deine letzte Instanz um solche Fragen zu beantworten ist der Standard:
http://www.open-std.org/jtc1/sc22/wg14/www/docs/n1570.pdfHier z.B. 7.21.5.3 The fopen function.
Also da steht:
w+b or wb+ truncate to zero lenght or create a binary file for updateWenn ich da weiter ein paar Zeilen weiter unten unter pos 7 nachsehe hat sorgt das "+" dafür, das da noch "besondere" Vorkehrungen für die Funktion erforderlich sind?
Also das update bedeutet nicht wie in der Zeile zuvor:
r+b Datei öffnen für update (lesen und schreiben)
sondern eine Sonderbehandlung damit sich der Stream in einem definierten Zustand befindet?
-
"r" : Öffnen zum Lesen. Datei muss vorhanden sein.
"r+" : Öffnen zum Lesen und Schreiben. Datei muss vorhanden sein.
"w" : Öffnen zum Schreiben. Datei wird neu erzeugt und ist leer (falls vorher bereits vorhanden: Inhalt geht verloren).
"w+" : Öffnen zum Lesen und Schreiben. Datei wird neu erzeugt und ist leer (falls vorher bereits vorhanden: Inhalt geht verloren).Mit zusätzlichem 'b' wird in allen Fällen der Binär-Modus festgelegt (keine Sonderbehandlung von Zeilentrennzeichen).
-
HalloC schrieb:
fwrite((char *)'\0', 1, size, rem);
Diese Zeile ist ja mal großer Müll, genauso wie der Rest vom Programm..
Gar nicht weiter darüber nachdenken, aus dem Hirn löschen.
-
Hier mal etwas ausführlicher zum Nachvollziehen, wie man testen könnte, um den ganzen Ungereimtheiten auf die Schliche zu kommen:
- Ich führe das Orginal-Programm aus und stelle fest, dass danach die test.txt nicht mehr vorhanden ist (soweit O.K.).
- Ich möchte mich vergewissern, dass das Überschreiben wirklich funktioniert, und kommentiere die remove(rm)-Zeile aus.
- Ich führe das geänderte Programm aus und stelle fest, dass die test.txt immer leer ist (Länge 0 Byte).
- Ich stelle fest (Debugger), dass fopen() hierfür verantwortlich ist und bemerke, dass für mode "r+b" statt "w+b" übergeben werden muss.
- Ich führe das geänderte Programm aus und es crasht.
- Ich stelle fest (Debugger), dass die Bufferadresse bei fwrite() NULL ist. Ich behelfe mich damit, dass ich bei fwrite() die Adresse eines gültigen Buffers mit vorerst nur einem 0-Byte übergebe [(char ""] und den Count-Parameter auf 1 (statt "size") setze.
- Ich führe das geänderte Programm aus und stelle fest, dass die Datei test.txt um ein 0-Byte am Ende größer wird, statt dass das erste Byte überschrieben wird.
- Ich schließe auf eine aktuell falsche Datei-Position und stelle fest, dass vor dem Schreiben mit fwrite mittels Einfügen von fseek(rem, 0L, SEEK_SET) wieder auf den Anfang postioniert werden muss.
- Ich führe das geänderte Programm aus und es wird wie erwartet das 1. Byte von test.txt mit einem 0-Byte überschrieben.Damit wären die gröbsten Bugs hoffentlich ausgemerzt. Jetzt muss noch folgendes erledigt werden:
- Das Überschreiben auf alle Bytes der Datei ausdehnen (für entsprechenden Buffer sorgen / Schleife).
- remove(rm) wieder aktivieren.Jetzt kannst du zur Übung vielleicht noch einige Erweiterungen vornehmen:
- Returncodes auswerten (fseek, ftell, fwrite, fclose, remove) sowie my_remove() mit Returncode ausstatten, der in main() ausgewertet werden kann.
- Kommadozeilenparameter für Dateiname, Löschzeichen, evtl. Option für Löschen oder Beibehalten (zwecks Test) der Datei nach dem Überschreiben.
- Ausgabe von Fehler-/Erfolgsmeldung.
-
my_remove schrieb:
Du brauchst in der jetzigen Form einen mit 0-Bytes initialisierten Buffer in der Größe der Dateilänge, oder auch, bei einem kleineren Buffer, eine Schleife, die der Dateilänge entsprechend viele 0-Bytes schreibt.
Der erste Vorschlag ist gut, der zweite eher nicht so, weil alles, was mit lesen und schreiben bzw. IO zu tun hat, unglaublich langsam ist wegen Umschaltung auf kernelmode, schreibkopf positionieren usw.
Daher besser alles in einen Buffer und mit einem Rutsch schreiben.
-
HansKlaus schrieb:
my_remove schrieb:
Du brauchst in der jetzigen Form einen mit 0-Bytes initialisierten Buffer in der Größe der Dateilänge, oder auch, bei einem kleineren Buffer, eine Schleife, die der Dateilänge entsprechend viele 0-Bytes schreibt.
Der erste Vorschlag ist gut, der zweite eher nicht so, weil alles, was mit lesen und schreiben bzw. IO zu tun hat, unglaublich langsam ist wegen Umschaltung auf kernelmode, schreibkopf positionieren usw.
Daher besser alles in einen Buffer und mit einem Rutsch schreiben.Du kannst dich auf Pufferung der crt und des os verlassen, das Rad neuerfinden ist nicht nötig.
-
naja aber allein dadurch, dass ich bei einer schleife geschätzte 10mio schleifenaufrufe im sinne von "schreibe das zeichen an der adresse x nach datei" (also nochmal zig zugriffe auf den speicher) habe, während es bei nem puffer vielleicht nur 10000 aufrufe im sinne von "schreibe alle zeichen in dem array an adresse x nach datei" sind.
jedenfalls war es mal eine laboraufgabe, genau dieses problem zu untersuchen.
-
HansKlaus schrieb:
... während es bei nem puffer vielleicht nur 10000 aufrufe im sinne von "schreibe alle zeichen in dem array an adresse x nach datei" sind. ...
Das ist aber genau das, was mein, wie du sagstest, 2. nicht so guter Vorschlag, beinhaltet, nämlich diesen "kleineren Buffer", der ja nicht zwingend nur ein Byte groß sein muss, aber bei einer Datei von erheblicher Größe bekomme ich möglicherweise keinen Buffer in der Größe der Dateilänge, wodurch wiederholtes Schreiben über eine geeignete Schleifenkonstruktion erforderlich wird.
In meiner Testumgebung war übrigens festzustellen, dass bei fwrite intern ein 4K grosser Buffer ins Spiel kommt.
-
ja ich hab das mit dem kleinen buffer überlesen und dachte, du meintest eine umsetzung mit fputc o.ä. sorry!
-
HalloC schrieb:
Es wird wohl Zeit für ein Literaturwechsel
MfG
TomNa dann mach mal.
http://www.ebay.de/itm/141905181959?_trksid=p2055119.m1438.l2649&ssPageName=STRK%3AMEBIDX%3AIT
MFG
Tom