Gestione dei File Corso di Informatica A Vito Perrone 1 Il file system in C Indice Principali operazioni sui file (dalla Standard Library) Esempi 2
Il file system in C Flusso di comunicazione: binario (sequenza di byte) di tipo testo (sequenza di caratteri) Modalità di utilizzo: Lettura, scrittura o lettura e scrittura; Posizione corrente: Punta al prossimo byte da leggere o scrivere sul file; Indicatore di errore; Indicatore di end-of-file (eof). 3 Variabili puntatore stdin stdout stderr f1 f2 f3 Tabella file aperti Tabella dei file aperti nomefile:. nomefile:. nomefile:. nomefile: FileTre. nomefile: FileUno. nomefile: FileDue. File StandardInput StandardInput StandardError FileUno FileDue FileTre 4
Operazioni di gestione dei file FILE *fopen (nomefile, modalità) int fclose (FILE *fp) int remove (nomefile) int rename (vecchionome, nuovonome) int ferror (FILE *fp) int feof (FILE *fp) void clearerr (FILE *fp) 5 Operazioni di lettura/scrittura int fprintf (FILE *fp, stringa di controllo, elementi) int fscanf (FILE *fp, stringa di controllo, indirizzo elementi) int getchar (void) int putchar (int c) int fgetc (FILE *fp) int fputc (int c, FILE *fp) 6
Esempi con file di testo (1) /*Legge e mostra sul video il contenuto del file di testo filechar*/ #include <stdio.h> /* Contiene la definizione di EOF, del tipo FILE e le testate delle funzioni che operano su file */ #include <stddef.h> /* Contiene la definizione di NULL */ main() FILE *fp; char c; if ((fp = fopen("filechar", "r"))!= NULL) /* Il file viene aperto in lettura con modalità testo */ while ((c = fgetc(fp))!= EOF) /* Viene letto e stampato un carattere per volta sino a fine file */ putchar(c); fclose(fp); else printf("il file non può essere aperto\n"); 7 Esempi con file di testo (2) #include <stdio.h> #include <stddef.h> #include <string.h> #define OK 1 #define ERROR 0 #define MAXLINE 100 int copiaselettiva(char refstr[]) char line[maxline]; FILE *fin, *fout; /*Lettura e scrittura di stringhe (1)*/ If ((fin = fopen("filein", "r")) == NULL) /* filein viene aperto in lettura con modalità testo */ return ERROR; if ((fout = fopen("fileout", "w")) == NULL) /* fileout viene aperto in scrittura con modalità testo */ fclose(fin); return ERROR; 8
Esempi con file di testo (3) /*Lettura e scrittura di stringhe (2)*/ while (fgets(line,maxline,fin)!= NULL) /* fgets legge da filein al più MAXLINE 1 caratteri e assegna al vettore line i caratteri letti, incluso l'eventuale carattere di newline, e termina la stringa con il carattere \0 */ if (strstr (line,refstr)!= NULL) /* La funzione strstr restituisce la posizione della prima occorrenza della stringa puntata da refstr nella stringa puntata da line; se la seconda stringa non è contenuta nella prima viene restituito il valore NULL */ fputs(line,fout); fclose(fin); fclose(fout); return OK; 9 Lettura e scrittura di strutture int fread(void *ptr, dimelemento, numelementi, FILE *fp); int fwrite(void *ptr, dimelemento, numelementi, FILE *fp); 10
Esempio con file binario (1) /*Gestione file Persone (1)*/ typedef struct char nome[20]; char cognome[20]; char indirizzo[50]; Persona; typedef char CodFisc[16]; typedef struct char nome[20]; char cognome[20]; char indirizzo[50]; CodFisc CodiceFiscale; NuovaPersona; /* I file Persone, CodiciFiscali e Nuove Persone si suppongono aperti dal main. pp, cf e np fanno riferimento ai tre file in questione */ int AggiornaPersone (FILE *pp, FILE *cf, FILE *np) Persona PersonaCorrente; CodFisc CodFiscCorrente; NuovaPersona NuovaPersonaCorrente; 11 Esempio con file binario (2) rewind(pp); rewind(cf); rewind(np); /*Gestione file Persone (2)*/ /* Rende possibile le seguenti operazioni di lettura e scrittura sul file identificato da pp, iniziando dal primo byte del file.*/ while (fread(&personacorrente,sizeof(persona),1,pp)!= 0) /* Finché non si è raggiunta la fine del file */ fread(&codfisccorrente,sizeof(codfisc),1,cf); strcpy(nuovapersonacorrente.nome, PersonaCorrente.nome); strcpy(nuovapersonacorrente.cognome,personacorrente.cognome); strcpy(nuovapersonacorrente.indirizzo, PersonaCorrente.indirizzo); strcpy(nuovapersonacorrente.codicefiscale, CodFiscCorrente); fwrite(&nuovapersonacorrente,sizeof(nuovapersona),1,np); 12
Accesso diretto (1) int fseek(file *fp, long offset, int refpoint) SEEK_SET : scostamento rispetto all inizio del file, SEEK_CUR : scostamento rispetto alla posizione corrente, SEEK_END : scostamento rispetto alla fine del file. long ftell(file *fp) rewind(f) equivale a: fseek (f, 0, SEEK_SET); 13 Accesso diretto (2)... /*Inversione del contenuto di un file numint di interi (1)*/ main () FILE *f; long int inizio, fine; int tempi, tempf; unsigned int dim; if ((f = fopen ("numint", "rb+")) == NULL) puts ("Non è stato possibile aprire il file numint"); /* più efficiente della printf per la stampa di un messaggio dato che non richiede la scansione e l interpretazione della stringa di controllo */ exit(1); /* La funzione exit provoca una conclusione non anomala del programma e la restituzione del controllo al sistema operativo */ 14
Accesso diretto (3) /*Inversione del contenuto di un file numint di interi (2)*/... inizio = 0; dim = sizeof(int); fseek (f, dim, SEEK_END); /* SEEK_END è una costante definita nel file stdio.h. Ha valore 2 */ fine = ftell (f); while (inizio < fine) fseek (f, inizio, SEEK_SET); fread (&tempi, dim, 1, f); fseek (f, fine, SEEK_SET); fread (&tempf, dim, 1, f); /* È necessario riposizionarsi */ fseek (f, fine, SEEK_SET); fwrite (&tempi, dim, 1, f); fseek (f, inizio, SEEK_SET); fwrite (&tempf, dim, 1, f); inizio = inizio + dim; fine = fine dim; 15 Programma Servizio Voli (1) /* Programma ServizioVoli (1)*/ void main () [dichiarazioni varie: esse dovranno definire le variabili necessarie per far riferimento ai file che memorizzano l'archivio, le varie funzioni destinate alla sua gestione in relazione alle operazioni richieste, le variabili e i tipi associati all'archivio e le variabili e tipi necessari per la realizzazione del menu] Fine = false; do [RipulisciLoSchermo] puts ("\nsono disponibili le seguenti operazioni. Per selezionare un'operazione premere il tasto indicato alla destra dell'operazione e il carattere di Invio.\n"); /* Ricordiamo che puts inserisce dopo la stringa un newline ottenendo l'andata a capo */ puts ("Prenotazione volo: P"); puts ("Cancellazione di una prenotazione: C"); puts ("Inserimento in lista di attesa: A"); puts ("Fine operazioni: F"); Risposta = getchar (); 16
/* Programma ServizioVoli (2)*/ Risposta = getchar (); switch (Risposta) case 'P': ServiPrenotazioni(); break; case 'C': ServiCancellazioni(); break; case 'A': ServiListaAttesa(); break; case 'F': Fine = true; break; while (Fine == false); [RipulisciLoSchermo] puts ("\n\n\narrivederci\n"); Programma Servizio Voli (2) 17 Programma Servizio Voli (3) /* Programma ServizioVoli (3)*/ #define MAXPOSTI 350 typedef struct char Cognome[30]; char Nome[30]; Passeggero; typedef struct int NumVolo; char Data[10]; Passeggero Prenotazioni[MAXPOSTI]; Passeggero ListaAttesa[100]; DescrizioneVolo; 18
Programma Servizio Voli (4) /* Programma ServizioVoli (4)*/ boolean char FILE Fine; Risposta; *av; /* Puntatore al descrittore del file ArchivioVoli */ void ServiPrenotazioni (void); /* Esegue, se possibile, la prenotazione richiesta. In caso contrario, chiede automaticamente all'utente se desidera essere messo in lista di attesa. Se la risposta è positiva, chiama la funzione ServiListaAttesa senza bisogno di ritornare al menu principale. Dopo l'esecuzione della funzione ServiListaAttesa, termina anche la ServiPrenotazioni. L'esecuzione della funzione comporta l'interazione con l'utente attraverso un opportuno menu in cui vengono chiesti all'utente il numero e la data del volo desiderato, il cognome e il nome del passeggero. Dopo aver ottenuto i dati richiesti la funzione accede all'archivio e provvede alle necessarie operazioni di interrogazione e aggiornamento */ 19 Programma Servizio Voli (5) /* Programma ServizioVoli (5)*/ void ServiCancellazioni (void); /* Vengono chiesti, tramite opportuno menu, i dati del passeggero e l'identificazione del volo al quale si intende rinunciare. Viene cancellato il passeggero dalla relativa lista. Inoltre, se la lista d'attesa del volo non è vuota, si estrae il suo primo elemento e lo si inserisce nella lista delle prenotazioni. Viene quindi scritto un messaggio sul terminale che avvisa l'operatore che un elemento della lista d'attesa ha ora la sua prenotazione confermata */ void ServiListaAttesa(void); /* Vengono chieste le solite informazioni e viene inserito il passeggero nella lista d'attesa desiderata */ 20