Es kann einen Leistungsunterschied geben oder auch nicht. Es gibt einen Unterschied in der Semantik.
fread(a, 1, 1000, stdin);
versucht, 1000 Datenelemente zu lesen, von denen jedes 1 Byte lang ist.
fread(a, 1000, 1, stdin);
versucht 1 Datenelement zu lesen, das 1000 Bytes lang ist.
Sie sind anders, weil fread()
die Anzahl der Datenelemente zurückgeben, die gelesen werden konnten, nicht die Anzahl der Bytes. Wenn es vor dem Lesen der vollen 1000 Bytes das Dateiende (oder eine Fehlerbedingung) erreicht, muss die erste Version genau angeben, wie viele Bytes es gelesen hat. Die zweite schlägt einfach fehl und gibt 0 zurück.
In der Praxis wird wahrscheinlich nur eine untergeordnete Funktion aufgerufen, die versucht, 1000 Bytes zu lesen, und angibt, wie viele Bytes tatsächlich gelesen werden. Bei größeren Lesevorgängen werden möglicherweise mehrere Anrufe auf niedrigerer Ebene getätigt. Die Berechnung des von zurückzugebenden Wertesfread()
ist unterschiedlich, aber der Aufwand für die Berechnung ist trivial.
Es kann einen Unterschied geben, wenn die Implementierung vor dem Versuch, die Daten zu lesen, feststellen kann, dass nicht genügend Daten zum Lesen vorhanden sind. Wenn Sie beispielsweise aus einer 900-Byte-Datei lesen, liest die erste Version alle 900 Byte und gibt 900 zurück, während die zweite möglicherweise nicht die Mühe macht, etwas zu lesen. In beiden Fällen wird die Dateipositionsanzeige um die Anzahl der Zeichen vorgerückt erfolgreich gelesenen , dh 900, .
Aber im Allgemeinen sollten Sie wahrscheinlich auswählen, wie Sie es aufrufen möchten, basierend auf den Informationen, die Sie daraus benötigen. Lesen Sie ein einzelnes Datenelement, wenn ein teilweises Lesen nicht besser ist, als überhaupt nichts zu lesen. Lesen Sie kleinere Blöcke ein, wenn Teillesungen hilfreich sind.
fread(a, 1000, N, stdin);
der fp-Indikator nicht immer um ein Vielfaches von1000
?ftell
vorfread
undfseek
nach dem Anruf an .Gemäß der Spezifikation können die beiden durch die Implementierung unterschiedlich behandelt werden.
Wenn Ihre Datei weniger als 1000 Bytes enthält
fread(a, 1, 1000, stdin)
(lesen Sie 1000 Elemente zu je 1 Byte), werden weiterhin alle Bytes bis EOF kopiert. Andererseits wird das Ergebnis vonfread(a, 1000, 1, stdin)
(1 1000-Byte-Element lesen) in gespeicherta
ist, nicht spezifiziert, da nicht genügend Daten vorhanden sind, um das Lesen des 'ersten' (und einzigen) 1000-Byte-Elements zu beenden.Natürlich können einige Implementierungen das 'partielle' Element immer noch in so viele Bytes wie nötig kopieren.
quelle
Das wäre ein Implementierungsdetail. In glibc sind die beiden in der Leistung identisch, da es im Wesentlichen wie folgt implementiert ist (Ref http://sourceware.org/git/?p=glibc.git;a=blob;f=libio/iofread.c ):
size_t fread (void* buf, size_t size, size_t count, FILE* f) { size_t bytes_requested = size * count; size_t bytes_read = read(f->fd, buf, bytes_requested); return bytes_read / size; }
Beachten Sie, dass die C.
und POSIXStandard garantiert nichtsize
, dass jedes Mal ein vollständiges Objekt der Größe gelesen werden muss. Wenn ein vollständiges Objekt nicht gelesen werden kann (z. B.stdin
nur 999 Byte, aber Sie haben es angefordertsize == 1000
), bleibt die Datei in einem unbestimmten Zustand (C99 §7.19.8.1 / 2).Bearbeiten: Siehe die anderen Antworten zu POSIX.
quelle
fread
Anrufegetc
intern. inMinix
Anzahl , wie oftgetc
genannt wird , ist einfachsize*nmemb
so , wie oftgetc
hängt von der aufgerufen wird Produkt dieser beiden. Also beidesfread(a, 1, 1000, stdin)
undfread(a, 1000, 1, stdin)
wirdgetc
1000=(1000*1)
mal laufen . Hier ist die einfache Implementierungfread
von Minixsize_t fread(void *ptr, size_t size, size_t nmemb, register FILE *stream){ register char *cp = ptr; register int c; size_t ndone = 0; register size_t s; if (size) while ( ndone < nmemb ) { s = size; do { if ((c = getc(stream)) != EOF) *cp++ = c; else return ndone; } while (--s); ndone++; } return ndone; }
quelle
Möglicherweise gibt es keinen Leistungsunterschied, aber diese Anrufe sind nicht gleich.
fread
Gibt die Anzahl der gelesenen Elemente zurück, sodass diese Aufrufe unterschiedliche Werte zurückgeben.Es gibt keinen großen Unterschied in der glibc-Implementierung , bei der nur die Elementgröße mit der Anzahl der Elemente multipliziert wird, um zu bestimmen, wie viele Bytes gelesen werden sollen, und die gelesene Menge durch die Elementgröße am Ende dividiert wird. Die Version mit einer Elementgröße von 1 gibt jedoch immer die richtige Anzahl gelesener Bytes an. Wenn Sie jedoch nur Elemente einer bestimmten Größe vollständig lesen möchten, können Sie mit dem anderen Formular keine Unterteilung vornehmen.
quelle
Ein weiterer Satz aus http://pubs.opengroup.org/onlinepubs/000095399/functions/fread.html ist bemerkenswert
Die Funktion fread () liest in das Array, auf das ptr zeigt, bis zu nitems-Elementen, deren Größe durch die Größe in Bytes angegeben wird, aus dem Stream, auf den der Stream zeigt. Für jedes Objekt müssen Größenaufrufe an die Funktion fgetc () durchgeführt und die Ergebnisse in der gelesenen Reihenfolge in einem Array von vorzeichenlosen Zeichen gespeichert werden , die genau über dem Objekt liegen.
In beiden Fällen wird in Kürze auf die Daten von fgetc () zugegriffen ...!
quelle
fgetc
, also ist Posix tatsächlich auf C99 ausgerichtet. Der Standard gibt einem konformen Programm jedoch keine Möglichkeit zu bestimmen, obfgetc
"wirklich" aufgerufen wird oder obfread
etwas anderes gleichwertig ist. 5.1.2.3 erklärt, dass der Standard nur das Verhalten einer "abstrakten Maschine" beschreibt und auflistet, auf welche Weise das tatsächliche Programm diesem Verhalten entsprechen muss. Dies wird in C ++ als "Als-ob" -Regel bezeichnet, aber nicht in C (mein Fehler früher). Nicht beobachtbares Verhalten muss nicht identisch sein.fgetc
aufgerufen wird (z. B. indem Sie Ihr Programm mit Ihrer eigenen Version dieser Funktion verknüpfen, z. B. indem Sie libc ändern und neu kompilieren), kann dies mit der Einschränkung geschehen dass die Funktion, die Sie ersetzen, nicht immer und nur aufgerufen wird, wenn der Standard die abstrakte Maschine als aufrufend beschreibt.fread
(oder einem anderen Bit C-Code) zu beschreiben. Es ist in Posix so dokumentiert, einfach weil es im Standard so dokumentiert ist.Ich wollte die Antworten hier klären. fread führt gepufferte E / A durch. Die tatsächlichen Leseblockgrößen, die Fread verwendet, werden durch die verwendete C-Implementierung bestimmt.
Alle modernen C-Bibliotheken haben mit den beiden Aufrufen die gleiche Leistung:
fread(a, 1, 1000, file); fread(a, 1000, 1, file);
Sogar so etwas wie:
for (int i=0; i<1000; i++) a[i] = fgetc(file)
Sollte zu denselben Festplattenzugriffsmustern führen, obwohl fgetc aufgrund von mehr Aufrufen in den Standard-c-Bibliotheken und in einigen Fällen der Notwendigkeit, dass eine Festplatte zusätzliche Suchvorgänge ausführt, die sonst wegoptimiert worden wären, langsamer wäre.
Zurück zum Unterschied zwischen den beiden Formen von Fread. Ersteres gibt die tatsächliche Anzahl der gelesenen Bytes zurück. Letzterer gibt 0 zurück, wenn die Dateigröße kleiner als 1000 ist, andernfalls gibt er 1 zurück. In beiden Fällen würde der Puffer mit denselben Daten gefüllt, dh dem Inhalt der Datei bis zu 1000 Byte.
Im Allgemeinen möchten Sie wahrscheinlich den 2. Parameter (Größe) auf 1 setzen, damit Sie die Anzahl der gelesenen Bytes erhalten.
quelle