|
|
|
![]() |
|
Strumenti |
![]() |
#1 |
Senior Member
Iscritto dal: Nov 2001
Città: Lainate (MI)
Messaggi: 831
|
[C] semplice programma r/w su file...
scusate ma questo semplice programma di scrittura/lettura di un file di interi mi duplica l'ultimo numero quando vado a leggerlo, sapreste dirmi perché?
Eppure sono arrivato al punto di copiarlo pari passo da un libro... ![]() Codice:
#include <stdio.h> #include <errno.h> #include <string.h> int main (int argc, char* argv[]){ int i; FILE* fPtr; char* errore; //valore stringa associato a errno //variabili locali fPtr = fopen("interi.txt", "w"); if (fPtr == NULL){ errore = strerror(errno); printf("Impossibile creare il file:\n %s\n", errore); return 1; //uscita con errore }//end if //inserimento dei valori nel file for (i = 0; i < 100; i++){ fwrite(&i, sizeof(int), 1, fPtr); }//end for fclose(fPtr); //chiudo file fPtr = fopen("interi.txt", "r"); //riapro in lettura if (fPtr == NULL){ errore = strerror(errno); printf("Impossibile creare il file:\n %s\n", errore); return 1; //uscita con errore }//end if //lettura dei dati inseriti while(!feof(fPtr)){ fread(&i, sizeof(int), 1, fPtr); printf("Letto numero: %d\n", i); }//end while fclose(fPtr); return 0; /*uscita corretta del programma*/ }//end main
__________________
Alea iacta est. Che io deceda se recedo |
![]() |
![]() |
![]() |
#2 |
Senior Member
Iscritto dal: Nov 2001
Città: Lainate (MI)
Messaggi: 831
|
up
__________________
Alea iacta est. Che io deceda se recedo |
![]() |
![]() |
![]() |
#3 | |
Senior Member
Iscritto dal: Nov 2005
Città: TO
Messaggi: 5206
|
Quote:
Poi la documentazione di feof() del VC++ dice: "The feof function returns a nonzero value after the first read operation that attempts to read past the end of the file." Vuol dire che feof ritorna nonzero (cioè file-del-file) solo dopo la prima lettura che cerca di leggere alla fine del file. Se il file è lungo 400 byte (come appunto il tuo caso specifico 4x100), quando hai letto tutti i 100 numeri e il file pointer è 400, feof non indica ancora eof. Quindi in pratica fai una lettura in più che però fallisce e non imposta la variabile 'i' (che rimane al valore precedente). Solo a quel punto feof() indica eof.
__________________
Andrea, SCJP 5 (91%) - SCWCD 5 (94%) |
|
![]() |
![]() |
![]() |
#4 |
Senior Member
Iscritto dal: Nov 2001
Città: Lainate (MI)
Messaggi: 831
|
Allora, ho provato il programma sia su windows (con dev-c++) che su linux (con gcc) con "wb"/"rb" e "r"/"w" e il risultato è lo stesso.
Andando a memoria quello che dici mi torna, cioè che ad ogni fread() viene incrementata di uno la posizione del puntatore del record del file. Però stando così le cose alla "prima" lettura dell'ultimo byte la posizione del puntatore al record "dovrebbe" andare oltre la fine del file e quindi far "scattare" feof(), cosa che in realtà non accade, e quindi rilegge. Inoltre il codice è molto simile a quello di un esempio tratto da un libro, ma quest'ultimo non indica nessun trucchetto da utilizzare per evitare questa duplicazione (magari che so, bisogna utilizzare una tecnica particolare che mi sfugge che non conosco essendo un neofita del C). L'unica differenza tra i due programmi è il fatto che nel libro si usa delle struct mentre io uso degli interi per far prima. Comunque ricontrollerò il testo, non si sa mai... Bisogna forse posizionare la "testina" con una particolare chiamata fseek()? P.S. Qualcuno sa se per caso è possibile visualizzare il valore del puntatore del record di un file con gdb?
__________________
Alea iacta est. Che io deceda se recedo |
![]() |
![]() |
![]() |
#5 |
Senior Member
Iscritto dal: Feb 2002
Messaggi: 906
|
//lettura dei dati inseriti
while(!feof(fPtr)){ fread(&i, sizeof(int), 1, fPtr); if(feof(fPtr)) break; printf("Letto numero: %d\n", i); }//end while |
![]() |
![]() |
![]() |
#6 | |
Senior Member
Iscritto dal: Nov 2001
Città: Lainate (MI)
Messaggi: 831
|
Quote:
ma è un "trick standard" di programmazione o è una tua soluzione a questo problema?
__________________
Alea iacta est. Che io deceda se recedo |
|
![]() |
![]() |
![]() |
#7 |
Senior Member
Iscritto dal: Nov 2005
Città: TO
Messaggi: 5206
|
Basterebbe anche solo:
while (fread (&i, sizeof(int), 1, fPtr) == 1) { .... } Il ciclo può terminare per causa di eof o di errore. Se dopo il ciclo si vuole testare se c'è stato errore, basta usare ferror().
__________________
Andrea, SCJP 5 (91%) - SCWCD 5 (94%) |
![]() |
![]() |
![]() |
#8 |
Senior Member
Iscritto dal: Nov 2001
Città: Lainate (MI)
Messaggi: 831
|
io continuo a non capire, se veramente bisogna leggere al di fuoi del file per avere l'eof non mispeigo perchè il libro non me lo segnali
![]()
__________________
Alea iacta est. Che io deceda se recedo |
![]() |
![]() |
![]() |
#9 |
Senior Member
Iscritto dal: Nov 2001
Città: Lainate (MI)
Messaggi: 831
|
notte
__________________
Alea iacta est. Che io deceda se recedo |
![]() |
![]() |
![]() |
#10 |
Senior Member
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
|
|
![]() |
![]() |
![]() |
#11 |
Senior Member
Iscritto dal: Nov 2001
Città: Lainate (MI)
Messaggi: 831
|
va bene allora guarderò un altro libro
__________________
Alea iacta est. Che io deceda se recedo |
![]() |
![]() |
![]() |
Strumenti | |
|
|
Tutti gli orari sono GMT +1. Ora sono le: 14:45.