File Binari (superiori)

lezione
lezione
File Binari (superiori)
Tipo di risorsa Tipo: lezione
Materia di appartenenza Materia: Informatica (istituti tecnici) per le superiori
Avanzamento Avanzamento: lezione completa al 100%

File Binari

modifica

Quando si vuole lavorare con i file binari non si utilizza la codifica ASCII , i dati contenuti nelle celle della ram che rappresentano delle variabili vengono scritti così come sono nel file,
 
per poi essere successivamente letti trasferendo la sequenza binaria dal file direttamente alle celle di memoria corrispondenti alle variabili del programma.
 

Aggiunta la libreria fstream ci troviamo 3 nuovi tipi di dato


  • ifstream che serve per gestire un file di input ( da cui possiamo leggere le informazioni e caricarle nelle variabili del programma)
per aprire un file bisogna specificare che è binario  (di default è di testo)   
con l'attributo ios::binary e allora il comando assume la forma
ifstream fi; fi.open("milan.dat", ios::binary)
  • ofstream che serve per gestire un file di output ( su cui possiamo scrivere le informazioni contenute nelle variabili del nostro programma.
per aprire un file bisogna specificare che è binario  (di default è di testo)  
con l'attributo ios::binary e allora il comando assume la forma
ifstream fi; fi.open("milan.dat", ios::binary)


  • fstream che serve per gestire un file di input/output (su cui possiamo sia leggere e sia scrivere delle informazioni) qui sono presenti due cursori distinti uno per le operazioni di lettura e uno per le operazioni di scrittura,
per aprirlo specificare i parametri binary, in, out come in questo esempio
fstream f;
f.open("inventario.dat",ios::binary|ios::in|ios::out);.

altri attributi possibili oltre a in,out e binary sono:

Attributo Significato
ios::ate (all'apertura gli indici di lettura e scrittura sono posizionati alla fine del file, se il file conteneva dei dati questi non vengono sovrascritti, gli indici lettura e scrittura possono essere riposizionati in altri punti del file dopo l'apertura)
ios::app (all'apertura l'indice di scrittura è posizionato alla fine del file e pronto solo per aggiungere ulteriori dati)
ios::trunc (elimina il contenuto del file che viene aperto)

che possono essere uniti usando l'operatore or bitwise rappresentato dal simbolo | quindi per un file binario a cui voglio aggiungere solo dei dati in coda

fo.open("milan.dat", ios::binary| ios::app)

Le operazioni che di solito si fanno su un file binario sono :

  • creare un file (vuoto) e scriverci una sequenza di dati corrispondenti ai valori di alcune variabili del nostro programma in C
  • aprire un file (esistente) e leggere i dati in sequenza (caricandoli in alcune variabili del programma in C)
  • aprire un file (esistente) e aggiungere in coda altre informazioni.
  • In un file binario è possible non solo leggere/scrivere i dati in sequenza ma anche posizionarsi all'interno del file (in un punto arbitrario) e leggere/scrivere i dati partendo da quel punto , sollevandoci così dall'onere di leggere/scrivere tutti i dati che lo precedono.
  • È possibile allora aprire un file esistente di input , posizionarsi in un punto specifico del file , prima dell'inizio della sequenza binaria di un specifica variabile e leggere solo quella specifica variabile.
  • È possibile allora aprire un file esistente di output , posizionarsi in un punto specifico del file , prima dell'inizio della sequenza binaria di un specifica variabile e sovrascriverla con una nuova sequenza binaria, che va a modificarne il valore.


  • È possibile allora aprire un file esistente di input/ e output , posizionarsi in un punto specifico del file , prima dell'inizio della sequenza binaria di un specifica variabile e sovrascrivere/leggere solo quel particolare dato, oppure possiamo aggiungere in coda delle nuove informazioni.

In un file binario di input/output ci sono due cursori che simbolicamente indichiamo con g e p , che indicano rispettivamente il punto di lettura e il punto di scrittura sul file. I comandi per leggere la posizione del punto di lettura è tellg( ), mentre quello di scrittura è tellp( ), entrambi .restituiscono la posizione rispetto all'inizio del file espressa in byte.


Per posizionarsi in un punto del file, bisogna esprimere la posizione fornendo un offset rispetto o all'inizio del file, o rispetto alla posizione corrente o rispetto alla fine del file. In pratica posizionati 50 byte dall'inizio del file, oppure posizionati -20 byte dalla fine del file, oppure posizionati 30 byte rispetto alla posizione corrente.


I simboli usati per esprimere rispetto a cosa misurare l'offset sono

ios::beg                     per indicare l'inizio del file
ios::end                     per indicare la fine del file
ios::cur                     per indicare la posizione corrente

mentre per esprimere l'offset basta un numero positivo o negativo che esprime lo spostamento in byte


per posizionarsi in un punto per la prossima lettura si usa il comando

seekg( offset, posizioneriferimento)

ad esempio seekg(40, ios::beg) significa 40 byte dopo l'inizio del file

ad esempio seekg(-23,ios::end) significa 23 byte prima della fine del file


analogamente per posizionarsi in un punto per la prossima scrittura si usa

seekp(offset, posizioneriferimento)


per scrivere un dato il comando è


write((char *)  indirizzovariabile,  dimensioneinbyte della variabile

per scrivere il valore di una variabile a di tipo intero (che occupa 4 byte) scriviamo


write((char *)  &a, 4)

se non conosciamo la dimensione della variabile possiamo usare la funzione

sizeof  nomevariabile

che ne restituisce la dimensione in byte, oppure

sizeof (int)

che restituisce la dimensione del tipo di dati intero, quindi possiamo anche scrivere

write((char *)  &a, sizeof a)
oppure
write((char *)  &a, sizeof (int) )

il cast (char *) permette di salvare i dati sul file come sequenze di byte (un char rappresenta un valore di un byte)

per scrivere un vettore vett di interi di dimensione 500 elementi, possiamo dare il seguente comando


write((char *)  vett, 500*sizeof( int))
oppure
write((char *)  vett, sizeof vett)

per leggere un dato il comando è

read((const char *) indirizzodovescrivereildato, dimensioneinbytedatodascrivere)

se abbiamo int a=12;
per scrivere il dato usiamo il comando

read((const char *)  &a,  sizeof a)

per leggere un vettore vett di interi di dimensione 500 elementi, possiamo dare il seguente comando


read((const char *)  vett, 500*sizeof( int))
oppure
read((const char *)  vett, sizeof vett)