Kapittel 12: Filer og strømmer. Java som første programmeringsspråk

Like dokumenter
Kapittel 12: Filer og strømmer INF 100. Java som første programmeringsspråk

Kapittel 12: Filer og strømmer. Java som første programmeringsspråk

Kapittel 12: Filer og strømmer. Java som første programmeringsspråk

Kapittel 14: Filer og strømmer. Kort versjon

Kapittel 14: Filer og strømmer

Kapittel 14: Filer og strømmer

Kapittel 14: Filer og strømmer. Kort versjon

Å lese tall fra en fil, klassen Scanner

Hittil har programmene kommunisert med omverden via tastatur og skjerm Ønskelig at data kan leve fra en kjøring til neste

INF1000-SIKT - Notat om I/O i Java

INF1010 våren 2018 tirsdag 23. januar

IN Notat om I/O i Java

Litt om pakker og mest om data inn og ut

Kapittel 9: Sortering og søking Kort versjon

INF Notat om I/O i Java

INF1010 våren 2019 Onsdag 30. januar. Mer om unntak i Java (med litt repetisjon av I/O først)

Kapittel 8: Sortering og søking

Enkel lesing og skriving i Java

TOD063 Datastrukturer og algoritmer

Kapittel 8: Sortering og søking INF100

Plan for dagen. Vprg 4. Dagens tema - filbehandling! Strømmer. Klassen FilLeser.java. Tekstfiler

Kapittel 8: Sortering og søking INF100

Forkurs INF1010. Dag 1. Andreas Færøvig Olsen Tuva Kristine Thoresen

Leksjon 7. Filer og unntak

Kapittel 9: Sortering og søking Kort versjon

Eksamen i emnet INF100 Grunnkurs i programmering (Programmering I) og i emnet INF100-F Objektorientert programmering i Java I Løsningsforslag

Kapittel 9: Sortering og søking Kort versjon

Kapittel 9: Sortering og søking Kort versjon

Kapittel 8: Programutvikling

INF1010 våren 2017 Onsdag 25. januar. Litt om unntak i Java

Eksamen i emnet INF100 Grunnkurs i programmering (Programmering I) og i emnet INF100-F Objektorientert programmering i Java I

Kapittel 15: Grafiske brukergrensesnitt: Enkel GUI. Del I

Leksjon 7. Filer og unntak

Kapittel 15: Grafiske brukergrensesnitt: Enkel GUI. Del I

Algoritmer og datastrukturer Vedlegg A.4 Filbehandling på char-nivå

Læringsmål for forelesningen

UNIVERSITETET I BERGEN Det matematisk-naturvitenskapelige fakultet

Eksamen i emnet INF100 Grunnkurs i programmering (Programmering I) Løsningsforslag

Lese fra fil. INF1000 : Forelesning 5. Eksempel. De vanligste lesemetodene. Metoder:

Algoritmer og datastrukturer A.1 Filbehandling på bit-nivå

Kapittel 7: Mer om arv

Kapittel 13: Grafiske brukergrensesnitt INF 100. Java som første programmeringsspråk

INF1000 EKSTRATILBUD. Stoff fra uke 1-5 (6) 3. oktober 2012 Siri Moe Jensen

Kapittel 5: Objektkommunikasjon

Løse reelle problemer

Kapittel 11: Unntakshåndtering. Java som første programmeringsspråk

INF 1010, vår 2005 Løsningsforslag uke 11

TDT4100 Objektorientert programmering

Hva er verdien til variabelen j etter at følgende kode er utført? int i, j; i = 5; j = 10; while ( i < j ) { i = i + 2; j = j - 1; }

Bruk av class Scanner, FileWriter og Formatter som alternativ til EasyIO i INF1000.

INF1010 våren Arv og subklasser del 1 pluss (hvis vi har tid) litt om Unntak, IO og Scanner-klassen

Løse reelle problemer

Kapittel 11: Unntakshåndtering. Java som første programmeringsspråk

Ta inn og ut av 2D-array. Java 6. Liste over ulike verdier i 2D-array. Det ferdige programmet. Vi skal lage et program som illustrerer hvordan man

Leksjon 7. Filer og unntak

Forelesningsquiz. Forelesning inf Java 5. Sett dere to (eller tre) sammen og besvar de fire spørsmålene på utdelt ark. Tid: 15 min.

Kapittel 1: Datamaskiner og programmeringsspråk

INF1010 våren Arv og subklasser del 1 (pluss litt I/O og unntaksbehandling)

Beskrivelse av programmeringsspråket Compila15 INF Kompilatorteknikk Våren 2015

INF1000: Forelesning 4. Mer om arrayer Metoder

IN 211 Programmeringsspråk. Java. på 20 enkle ark. spesielt for de som kan. Simula. (og gjerne litt C) Ark 1 av 20

Forelesning inf Java 4

Innhold. INF1000 Høst Klasser og objekter. Uke 7: Mer objektorientert programmering Siri Moe Jensen

programeksempel Et større En større problemstilling Plan for forelesingen Problemstillingen (en tekstfil) inneholdt ordet "TGA"

i=0 Repetisjon: arrayer Forelesning inf Java 4 Repetisjon: nesting av løkker Repetisjon: nesting av løkker 0*0 0*2 0*3 0*1 0*4

INF1000 : Forelesning 5

Oversikt. INF1000 Uke 1 time 2. Repetisjon - Introduksjon. Repetisjon - Program

Innhold uke 4. INF 1000 høsten 2011 Uke 4: 13. september. Deklarasjon av peker og opprettelse av arrayobjektet. Representasjon av array i Java

3 emner i dag! INF1000 Uke 5. Objekter og pekere. null. Litt om objekter, pekere og null Filer og easyio Litt mer om tekster

Fra Python til Java, del 2

Post-it spørsmål fra timen (Arv og subklasser)

Algoritmer og datastrukturer A.1 BitInputStream

INF Løsning på seminaropppgaver til uke 8

Oblig 4Hybelhus litt mer tips enn i oppgaven

Spørsmål fra forrige forelesning. INF1000 Forelesning 7. Oppførselen til inword()/inint()/etc. Operator-presedens i Java

Bruk av class Scanner, split(), FileWriter og Formatter som alternativ til easyio i INF1000.

MER OM ARRAYER. INF1000: Forelesning 4. Anta at vi ønsker å lagre en liste med navnene på alle INF1000-studentene:

Forkurs INF1010. Dag 3. Andreas Færøvig Olsen Gard Inge Rosvold Institutt for Informatikk, 15.

Mål. Pensum. TDT4110 Informasjonsteknologi grunnkurs: Tema: Filer og unntak (exceptions) Utgave 3: Kap. 6. Terje Rydland - IDI/NTNU

import java.io.*; import java.util.*; import javagently.text;

INF1010 våren Generalisering -spesialisering Gjenbruk av klasser. Ved arv. Klasse-hierarkier. Stein Gjessing.

Kapittel 6: Arv. Redigert av: Khalid Azim Mughal

Løsningsforslag Test 2

EKSAMEN 6108/6108N PROGRAMMERING I JAVA Alt trykt og skriftlig materiale.

Tre måter å lese fra terminal. Java 4. Eksempel. Formatert utskrift til skjerm

UNIVERSITETET I OSLO

Fra Python til Java. En introduksjon til programmeringsspråkenes verden. Dag Langmyhr

Algoritmer og datastrukturer Kapittel 11 - Delkapittel 11.2

Ole Christian Lingjærde, 12. september 2013

Forelesning inf Java 4

1. Finn klassene (hvilke objekter er det i problemet) 1. Dataene som beskriver problemet (hvilke objekter har vi og hvor mange klasser er det?

Forkurs INF1010. Dag 3. Andreas Færøvig Olsen Eivind Storm Aarnæs

INF 1000 høsten 2011 Uke september

Algoritmer og datastrukturer Vedlegg A.3 Filer på byte-nivå

Forelesning inf Java 5

I dag. Rep: Oppsummering - variabler. Rep: Datatyper. INF1000 (Uke 3) Mer om uttrykk, terminal I/O, forgreninger

Forelesning inf Java 5

INF1000 undervisningen INF 1000 høsten 2011 Uke september

INF1000 (Uke 5) Mer om løkker, arrayer og metoder

Mer objektorientert programmering

Transkript:

Kapittel 12: Filer og strømmer Forelesningsnotater for: Java som første programmeringsspråk Khalid Azim Mughal, Torill Hamre, Rolf W. Rasmussen Cappelen Akademisk Forlag, 2003. ISBN 82-02-23274-0 http://www.ii.uib.no/~khalid/jfps/ Vi gir tillatelse til å bruke disse notatene i tilknytning til boken. Modifisert: 14/4/05 JFPS 12: Filer og strømmer 12-1/52

Emneoversikt Strømmer Filer Dataposter Binære filer Tekstfiler Terminal I/O Objektserialisering Direkte filtilgang JFPS 12: Filer og strømmer 12-2/52

Inn- og ut-data Inndata: Data som et program leser utenfra. Inndata kommer fra en kilde som produserer dataene. F.eks. tastatur, en fil Utdata: Data som et program skriver ut Utdata skrives ut til et mål som kan motta dataene. F.eks. skjerm, en fil En datafil tilbyr permanent lagring av data fra et program på et eksternt medium. JFPS 12: Filer og strømmer 12-3/52

Strømmer En strøm er et objekt som et program kan bruke til å skrive data til et mål (ut-strøm) eller lese data fra en kilde (inn-strøm). Sekvensielle strømmer: Data bare kan leses eller skrives med én verdi om gangen, dvs. som en sekvens. JFPS 12: Filer og strømmer 12-4/52

Bytestrømmer og tegnstrømmer Bytestrømmer håndterer sekvenser av bytes. Data bestående av åtte bits. Tegnstrømmer: håndterer sekvenser av tegn. Data bestående av 16-bits Unicode-tegn. JFPS 12: Filer og strømmer 12-5/52

Bytestrømmer (Figur 12.1) java.io InputStream FilterInputStream... DataInputStream «interface» DataInput ObjectInputStream «interface» ObjectInput FileInputStream... OutputStream FilterOutputStream DataOutputStream «interface» DataOutput PrintStream... ObjectOutputStream «interface» ObjectOutput... FileOutputStream JFPS 12: Filer og strømmer 12-6/52

Tegnstrømmer (Figur 12.2) java.io Reader... BufferedReader InputStreamReader... FileReader Writer... BufferedWriter OutputStreamReader PrintWriter FileWriter JFPS 12: Filer og strømmer 12-7/52

Filbehandling En fil betegner et spesifikt lagerområde på et eksternt medium, for eksempel harddisk, der informasjon er lagret. Data i filer blir lagret som en sekvens av bytes. Binærfil: Data tolket som en sekvens av bytes. Tekstfil: Data tolket som en sekvens av tegn. JFPS 12: Filer og strømmer 12-8/52

Standard lengde for predefinerte datatyper (Tabell 12.1) Predefinerte typer Antall bytes boolean 1 char 2 int 4 long 8 float 4 double 8 JFPS 12: Filer og strømmer 12-9/52

Filsti En filsti identifiserer en fil i filsystemet: String datafilnavn = "ansattfil.dat"; // Filsti angir filen. String filsti1 = "firma\\ansattfil.dat"; // Windows String filsti2 = "firma/ansattfil.dat"; // Unix String filnavn = "firma" + File.separator + "ansattfil.dat"; // Plattformuavhengig JFPS 12: Filer og strømmer 12-10/52

Dataposter (Figur 12.3) Problemstilling: lagre opplysninger om ansatte (Program 12.1) på en fil. String fornavn; String etternavn; double timelønn; boolean kjønn; En post er en oppstilling av ett eller flere datafelt. Et felt består vanligvis av en primitiv verdi, men en streng kan inngå som en feltverdi. felt 1 felt 2 felt 3 felt 3 "Ole" "Olsen" 215.0 false post felt 1 felt 2 felt 3 felt 3 "Bill" "Bailey" 910.0 false JFPS 12: Filer og strømmer 12-11/52

Binære filer En binærfil lagrer binære representasjoner av primitive verdier definert i Java. Tallverdier på filen opptar lagerplass i henhold til lengden definert i Java-programmeringsspråket. Tegn blir representert i Unicode-tegnsettet på i en binærfil. De boolske verdiene true og false blir lagret på filen med binær representasjon gitt ved henholdvis (byte)1 og (byte)0. Programmet må sørge for at verdiene som skrives ut blir tolket riktig ved innlesing. JFPS 12: Filer og strømmer 12-12/52

It's now or never Flaming Star Suspious Minds Angel Crying in the Chapel If I can dream Fever CC Rider Little Sister Are you lonesome tonight? Devil in desguise Utskriving av binære verdier (Figur 12.4) Objekt av klassen DataOutputStream Objekt av klassen FileOutputStream writeboolean(boolean v) writechar(int v) writechars(string s) writedouble(double v) writefloat(float v) writeint(int v) writelong(long v)... (2) (1) binærfil FileOutputStream utfilstrøm = new FileOutputStream(binærfilnavn, utvid); // (1) DataOutputStream utbinstrøm = new DataOutputStream(utFilStrøm); // (2) JFPS 12: Filer og strømmer 12-13/52

Saksgangen for å skrive binære verdier på en fil (Figur 12.4, Program 12.2) 1. Definer et FileOutputStream-objekt som åpner filen det skal skrives til: FileOutputStream utfilstrøm = new FileOutputStream(datafilnavn, utvid); 2. Definer et DataOutputStream-objekt som kobles til FileOutputStream-objekttet: DataOutputStream utbinstrøm = new DataOutputStream(utFilStrøm); 3. Skriv verdier binært med relevante writex()-metoder definert i DataOutput- Stream-klassen. void skrivansattdata(dataoutputstream utbinstrøm) // (13) throws IOException { // Et strengfelt avsluttes med FELT_SLUTT_TEGN. utbinstrøm.writechars(fornavn + FELT_SLUTT_TEGN); utbinstrøm.writechars(etternavn + FELT_SLUTT_TEGN); utbinstrøm.writedouble(timelønn); utbinstrøm.writeboolean(kjønn); } 4. Avslutt med å lukke strømmen for å frigjøre filen og andre ressurser: utbinstrøm.close(); JFPS 12: Filer og strømmer 12-14/52

Eksempel: Skriving av binære verdier (Program 12.2) Metoden skrivansattinfotilbinfil(), definert i klassen PersonellRegisterBin, skriver ut opplysninger om et ansatt. // Opprett strømmer for binærskriving. (4) FileOutputStream utfilstrøm = new FileOutputStream(datafilnavn, utvid); DataOutputStream utbinstrøm = new DataOutputStream(utFilStrøm); // Skriv hvor mange poster (dvs. antall ansatte) som skal // lagres i filen. utbinstrøm.writeint(antallansatte); // (5) // Skriv ut informasjon om alle ansatte. for (int i = 0; i < antallansatte; i++) ansatttabell[i].skrivansattdata(utbinstrøm); // (6) utbinstrøm.close(); // (7) JFPS 12: Filer og strømmer 12-15/52

It's now or never Flaming Star Suspious Minds Angel Crying in the Chapel If I can dream Fever CC Rider Little Sister Are you lonesome tonight? Devil in desguise Innlesing av binære verdier (Figur 12.5) Objekt av klassen FileInputStream Objekt av klassen DataInputStream binærfil (1) (2) readboolean() readbyte() readchar() readdouble() readfloat() readint() readlong() readshort()... FileInputStream innfilstrøm = new FileInputStream(binærfilnavn); // (1) DataInputStream innbinstrøm = new DataInputStream(innFilStrøm); // (2) JFPS 12: Filer og strømmer 12-16/52

Innlesing av binære verdier Prosedyren for å lese binære verdier fra en fil er følgende (se Figur 12.5 og Program 12.2): 1. Definer et FileInputStream-objekt som åpner filen det skal leses fra: FileInputStream innfilstrøm = new FileInputStream(datafilnavn); 2. Definer et DataInputStream-objekt som kobles til FileInputStream-objektet: DataInputStream innbinstrøm = new DataInputStream(innFilStrøm); 3. Les verdier binært med relevante readx()-metoder definert i DataInputStreamklassen. timelønn = innbinstrøm.readdouble(); kjønn = innbinstrøm.readboolean(); 4. Avslutt ved å lukke strømmen: innbinstrøm.close(); JFPS 12: Filer og strømmer 12-17/52

Eksempel: Lesing av binære verdier (Program 12.2) Metoden lesansattinfofrabinfil() i klassen PersonellRegisterBin sørger for innlesing av binære verdier for et ansatt: // Opprett strømmer for binærinnlesing. (9) FileInputStream innfilstrøm = new FileInputStream(datafilnavn); DataInputStream innbinstrøm = new DataInputStream(innFilStrøm); // Les hvor mange poster som er i filen. (10) int totalansatte = innbinstrøm.readint(); // Les inn informasjon om hver ansatt. (11) for (int i = 0; i < totalansatte; i++) { // Opprett en "tom" ansatt. AnsattBin nyansatt = new AnsattBin(); // Les data i den. nyansatt.lesansattdata(innbinstrøm); // Innsett i registeret. leggtilansatt(nyansatt); } innbinstrøm.close(); // (12) JFPS 12: Filer og strømmer 12-18/52

Opplysninger om en ansatt blir lest av metoden lesansattdata() i klassen Ansatt- Bin, (14): fornavn = lesbinstrengfelt(innbinstrøm); etternavn = lesbinstrengfelt(innbinstrøm); timelønn = innbinstrøm.readdouble(); kjønn = innbinstrøm.readboolean(); Metoden lesbinstrengfelt() sørger for å lese en streng, (15): String str = ""; while(true) { char tegn = innbinstrøm.readchar(); if (tegn == FELT_SLUTT_TEGN) break; else str += tegn; } JFPS 12: Filer og strømmer 12-19/52

Håndtering av slutt på inndata for binær innlesing (Program 12.3) Slutt på inndata håndteres ved å fange kontrollerte unntak av typen EOFException. Metoden main() sørger for å opprette strømmene og behandle IOExceptions som metoden lesfrabinfil() kaster videre. JFPS 12: Filer og strømmer 12-20/52

Håndtering av slutt på inndata ved binær lesing (Program 12.3) import java.io.*; public class EOFUnntak { public static void main(string[] args) { // (1) try { lesfrabinfil(args[0]); } catch (FileNotFoundException unntak) { System.out.print("Filen ikke funnet: " + unntak); unntak.printstacktrace(); } catch (IOException unntak) { System.out.print("Feil ved lesing: " + unntak); unntak.printstacktrace(); } } JFPS 12: Filer og strømmer 12-21/52

} public static void lesfrabinfil(string filnavn) throws IOException { // (2) } FileInputStream innfilstrøm = new FileInputStream(filnavn); // (3) DataInputStream innbinstrøm = new DataInputStream(innFilStrøm); try { // (4) while (true) { // (5) int i = innbinstrøm.readint(); System.out.println(i); } } catch (EOFException unntak) { // (6) System.out.print("Slutt på inndata."); } innbinstrøm.close(); // (7) Utføring av programmet: >java EOFUnntak testdata.dat 2000 2001 2002 2003 2004 Slutt på inndata. JFPS 12: Filer og strømmer 12-22/52

Tekstfiler En tekstfil består av tekstlinjer. En tekstlinje består av en sekvens av tegn avsluttet med en linjeslutt-streng. Metodene i Java sørger for at denne strengen blir tolket riktig. Løsningen for å håndtere tegn mellom Java-program og omverdenen: Bytes som er lest av byte-inn-strømmen blir oversatt til Unicode-tegn av tegn-innstrømmen. Unicode-tegn (som skal skrives) oversettes til bytes av tegn-ut-strømmen og blir skrevet ut av byte-ut-strømmen. JFPS 12: Filer og strømmer 12-23/52

It's now or never Flaming Star Suspious Minds Angel Crying in the Chapel If I can dream Fever CC Rider Little Sister Are you lonesome tonight? Devil in desguise Utskrift til en tekstfil (Figur 12.6) print(boolean b) print(char t) print(char[] ttab) print(double d) print(float f) print(int i) print(long l) print(object obj) print(string str) println()... Objekt av klassen PrintWriter (2) (1) Objekt av klassen FileWriter tekstfil FileWriter tekstfilskriver = new FileWriter(tekstfilnavn,utvid); // (1) PrintWriter tekstskriver = new PrintWriter(tekstFilSkriver); // (2) JFPS 12: Filer og strømmer 12-24/52

Utskrift til tekstfiler (Figur 12.6 og Program 12.4) 1. Definer et FileWriter-objekt som åpner filen det skal skrives til: FileWriter tekstfilskriver = new FileWriter(datafilnavn, utvid); 2. Definer et PrintWriter-objekt som kobles til FileWriter-objektet: PrintWriter tekstskriver = new PrintWriter(tekstFilSkriver); 3. Skriv verdier ut som tekst med print()-metoder definert i PrintWriter- klassen. void skrivansattdata(printwriter tekstskriver) { // (13) // Felt adskilt med et feltslutt-tegn. tekstskriver.print(fornavn + FELT_SLUTT_TEGN); // (14) tekstskriver.print(etternavn + FELT_SLUTT_TEGN); // (15) tekstskriver.print(timelønn); // (16) tekstskriver.print(felt_slutt_tegn); // (17) tekstskriver.println(kjønn); // (18) } jfr. kildekoden for metoden skrivansattdata2(). 4. Avslutt ved å lukke strømmen: tekstskriver.close(); JFPS 12: Filer og strømmer 12-25/52

It's now or never Flaming Star Suspious Minds Angel Crying in the Chapel If I can dream Fever CC Rider Little Sister Are you lonesome tonight? Devil in desguise Innlesing fra en tekstfil (Figur 12.7) Objekt av klassen FileReader Objekt av klassen BufferedReader readline() read() tekstfil (1) (2) FileReader tekstfilleser = new FileReader(tekstfilnavn); BufferedReader tekstleser = new BufferedReader(tekstFilLeser); JFPS 12: Filer og strømmer 12-26/52

Innlesing fra tekstfiler (Figur 12.7, Figur 12.8 og Program 12.4): 1. Definer et FileReader-objekt som åpner filen det skal lese fra: FileReader tekstfilleser = new FileReader(datafilnavn); 2. Definer et BufferedReader-objekt som kobles til FileReader-objektet: BufferedReader tekstleser = new BufferedReader(tekstFilLeser); 3. Les én linje om gangen med readline()-metoden som returnerer et String-objekt: String post = tekstleser.readline(); Tegn i et felt kan hentes ut (steg 1 og 2 i Figur 12.8): int feltsluttindeks1 = post.indexof(felt_slutt_tegn); fornavn = post.substring(0, feltsluttindeks1); Felttegn må konverteres til tilsvarende primitiv verdi (steg 3 i Figur 12.8): String doublestr = post.substring(feltsluttindeks2 + 1, feltsluttindeks3); timelønn = Double.parseDouble(doubleStr); // (22) String sannhetsverdi = post.substring(feltsluttindeks3 + 1); kjønn = sannhetsverdi.equals("true"); // (23) jfr. kildekoden for metoden lesansattdata2(). 4. Avslutt ved å lukke strømmen. JFPS 12: Filer og strømmer 12-27/52

Innlesing og konvertering av tekst til primitive verdier (Figur 12.8) Steg 1: Finn indeks til felt-avslutt-tegn feltsluttindeks1 feltsluttindeks2 feltsluttindeks3 indeks Tekstlinje Steg 2: Ekstraher substreng 0 3 9 15 O l e O l s e n 2 1 5. 0 f a l s e \n felt 1 felt 2 felt 3 felt 3 "Ole" "Olsen" "215.0" "false" Steg 3: Konverter substreng til flyttall til boolsk verdi 215.0 false JFPS 12: Filer og strømmer 12-28/52

Terminal I/O Hvert Java-program under utføring får automatisk knyttet til seg følgende tre bytestrømmer: 1. System.in leser bytes fra tastaturet. 2. System.out skriver bytes til skjermen. 3. System.err for å skrive feilmeldinger til skjermen. Terminal er en utstyrsenhet som har et tastatur og en skjerm. JFPS 12: Filer og strømmer 12-29/52

Utskrift til terminalen System.out er et objekt av Klassen PrintStream, og brukes til å sende utskrift til skjermen (Figur 12.9): System.out.println("Livet er for kort. Spis desserten først!"); print(boolean b) print(char t) print(char[] ttab) print(double d) print(float f) print(int i) print(long l) print(object obj) print(string str) println() println(...)... Objekt av klassen PrintStream: System.out Terminal It's now or never Flaming Star Suspious Minds Angel Skjerm JFPS 12: Filer og strømmer 12-30/52

Avsøking med java.util.scanner-klassen En scanner konverterer inndata fra dens kilde til ord (eng. tokens). Whitespace (mellomrom, tabulatortegn, linjeskift) skiller ordene. Kilden kan være en String, en InputStream eller en Reader, og sendes som parameter i konstruktørkallet. Lookahead metoder boolean hasnext() Returnerer true dersom det er ord igjen inndata. boolean hasnextint() boolean hasnextlong() boolean hasnextdouble() boolean hasnextboolean() Returner true dersom neste ord i inndata kan tolkes som typen som tilsvarer metodenavnet. Analyse av ord i inndata String next() Returnerer neste ord i inndata. String nextline() Flytter innlesing forbi innværende linje, og returnerer inndata som ble hoppet over. int nextint() long nextlong() double nextdouble() boolean nextboolean() Tolker neste ord i inndata som typen som tilsvarer metodenavnet. void close() Lukker denne scanneren og evt. dens kilde. JFPS 12: Filer og strømmer 12-31/52

Innlesing fra terminalen Objekt av klassen InputStream: System.in Terminal Objekt av klassen Scanner It's now or never Flaming Star Suspious Minds Angel Tastatur (1) int nextint() long nextlong() double nextdouble() boolean nextboolean() String next() String nextline()... Scanner innleser = new Scanner(System.in); // (1) System.in er en byte-inn-strøm. Et Scanner-objekt kan kobles til System.in for innlesning. Metoder i Scanner-klassen brukes til å lese verdier som blir tastet. JFPS 12: Filer og strømmer 12-32/52

Eksempel: Innlesing fra terminalen import java.util.arraylist; import java.util.list; import java.util.scanner; import static java.lang.system.out; public class LesingFraTerminalMedScanner { public static void main(string[] args) { // En Scanner kobles til System.in, dvs. til terminalen. Scanner innleser = new Scanner(System.in); // Leser navn. String fornavn; String etternavn; String navn; String svar; do { System.out.print("Skriv fornavn: "); fornavn = innleser.next(); innleser.nextline(); // tøm inneværende linje for tegn. JFPS 12: Filer og strømmer 12-33/52

System.out.print("Skriv etternavn: "); etternavn = innleser.next(); innleser.nextline(); navn = etternavn + " " + fornavn; System.out.println("Navn: " + navn); System.out.print("Ett til? (j/n): "); svar = innleser.next(); } while (svar.equals("j")); } } // Leser en list med heltall. List<Integer> intlist = new ArrayList<Integer>(); out.println("skriv en liste med heltall," + " avslutt med en verdi som ikke er heltall:"); while(innleser.hasnextint()) { int i = innleser.nextint(); intlist.add(i); } for (int i : intlist) out.println(i); // Lukk scanner. innleser.close(); JFPS 12: Filer og strømmer 12-34/52

Kjøring av programmet: Skriv fornavn: Junior Skriv etternavn: Java Navn: Java Junior Ett til? (j/n): n Skriv en liste med heltall, avslutt med en verdi som ikke er heltall: 132 454 43-1 slutt 132 454 43-1 JFPS 12: Filer og strømmer 12-35/52

Klassen Terminal Klassen Terminal definerer statiske metoder for innlesning av verdier. public final class Terminal { private Terminal() {}; private static Scanner leser = new Scanner(System.in); public static int lesint() { while (true) try { return leser.nextint(); } catch (InputMismatchException ime) { reporterror(); } catch (NoSuchElementException nsee) { reporterror(); } } //... private static void reporterror() { leser.nextline(); // Tøm linjen først. System.out.println( "Feil i inndata. Prøv på nytt!" ); } } Metoden nextint() kan kaste to ikke-kontrollerte unntak som kan være en god idé å håndtere. JFPS 12: Filer og strømmer 12-36/52

Objektserialisering Prosessen for å skrive ut et objekt kalles serialisering. All relevant informasjon om et objekt skrives ut som en bytesekvens. Prosessen for å lese inn disse opplysningene for å gjenskape et objekt kalles deserialisering. Objektet kan "gjenskapes" ved å tolke dets bytesekvens. Klassene ObjectOutputStream og ObjectInputStream definerer henholdvis objektut-strømmer og objekt-inn-strømmer som tilbyr metoder for å serialisere og deserialisere objekter. Klassene kan også brukes til å skrive og lese binære verdier og objekter. For at et objekt til en klasse skal kunne serialiseres, må klassen implementere java.lang.serializable-kontrakten. interface Serializable {} // Tom kropp. class Ansatt implements Serializable {...} JFPS 12: Filer og strømmer 12-37/52

It's now or never Flaming Star Suspious Minds Angel Crying in the Chapel If I can dream Fever CC Rider Little Sister Are you lonesome tonight? Devil in desguise Utskriving av objekter og binære verdier (Figur 12.11) writeobject(object obj) Objekt av klassen ObjectOutputStream Objekt av klassen FileOutputStream writeboolean(boolean v) writechar(int v) writechars(string s) writedouble(double v) writefloat(float v) writeint(int v) writelong(long v)... (2) (1) datafil FileOutputStream utfilstrøm = new FileOutputStream(datafilnavn, utvid); // (1) ObjectOutputStream utobjstrøm = new ObjectOutputStream(utFilStrøm); // (2) JFPS 12: Filer og strømmer 12-38/52

Utskriving av objekter Metoden writeobject() i klassen ObjectOutputStream brukes til å skrive objekter. utobjstrøm.writeobject(nyansatt); Metoden skriver ut tilstanden til et objekt, som utgjør verdiene til feltvariabler i objektet. Statiske datamedlemmer blir ikke tatt med. Følgende metodekall serialisere hele personellregisteret: utobjstrøm.writeobject(personellregister); Følgende instans-datamedlemmer blir inkludert i denne serialiseringen: Ansatt[] ansatttabell; int antallansatte; int antallkvinner; Legg merke til at kallet ovenfor fører til at hele ansatttabellen også blir serialisert, inklusive Ansatt-objekter i tabellen. Ved innlesingen blir samme tilstand gjenskapt som før utskrivingen. JFPS 12: Filer og strømmer 12-39/52

Saksgangen for å serialisere objekter til en fil (Figur 12.11 og Program 12.6): 1. Definer et FileOutputStream-objekt som åpner filen det skal skrives til: FileOutputStream utfilstrøm = new FileOutputStream(datafilnavn, utvid); 2. Definer et ObjectOutputStream-objekt som kobles til FileOutputStream-objektet: ObjectOutputStream utobjstrøm = new ObjectOutputStream(utFilStrøm); 3. Skriv ut objekter med writeobject()-metoden: utobjstrøm.writeobject(ansatttabell); // (5) Skriver ut en tabell I tillegg til å skrive objekter kan vi også skrive primitive verdier, for eksempel: utobjstrøm.writeint(antallansatte); utobjstrøm.writeint(antallkvinner); 4. Avslutt ved å lukke strømmen: utobjstrøm.close(); JFPS 12: Filer og strømmer 12-40/52

It's now or never Flaming Star Suspious Minds Angel Crying in the Chapel If I can dream Fever CC Rider Little Sister Are you lonesome tonight? Devil in desguise Innlesing av objekter og binære verdier (Figur 12.12) Objekt av klassen FileInputStream Objekt av klassen ObjectInputStream readobject() datafil (1) (2) readboolean() readchar() readdouble() readfloat() readint() readlong()... FileInputStream innfilstrøm = new FileInputStream(datafilnavn); // (1) ObjectInputStream innobjstrøm = new ObjectInputStream(innFilStrøm); // (2) JFPS 12: Filer og strømmer 12-41/52

Innlesing av objekter Under deserialiseringen må man holde rede på at riktig antall objekter blir lest i riktig rekkefølge. Metoden readobject() i ObjectInputStream-klassen brukes til å lese inn ett serialisert objekt om gangen, inklusive dets delobjekter: Object objref = innobjstrøm.readobject(); Programmet må selv holde rede på hvilken klasse objektet egentlig tilhører: if (objref instanceof Ansatt) { // (1) Ansatt enansatt = (Ansatt) objref; // (2) double lønn = enansatt.beregnukelønn(47.5); // (3) } Metoden readobject() gjenskaper hele objektet: Object ref = innobjstrøm.readobject(); if (ref instanceof PersonellRegister) { PersonellRegister nyttpersonellregister = (PersonellRegister) ref; nyttpersonellregister.skrivansattinfotilterminal(); } JFPS 12: Filer og strømmer 12-42/52

Saksgangen for å deserialisere objekter fra en fil (Figur 12.12 og Program 12.6) 1. Definer et FileInputStream-objekt som åpner filen det skal leses fra: FileInputStream innfilstrøm = new FileInputStream(datafilnavn); 2. Definer et ObjectInputStream-objekt som kobles til FileInputStream-objektet: ObjectInputStream innobjstrøm = new ObjectInputStream(innFilStrøm); 3. Les et objekt med readobject()-metoden. Kontroller objektets type før vi kaller metoder definert spesifikt i objektets klasse. // Les et objekt fra filen. Object nyttobj = innobjstrøm.readobject(); // (9) // Kontroller at det er en ansatttabell. if (nyttobj instanceof Ansatt[]) ansatttabell = (Ansatt[]) nyttobj; // (10) else System.out.println("Ansatttabell ikke funnet."; 4. Avslutt ved å lukke strømmen: innobjstrøm.close(); JFPS 12: Filer og strømmer 12-43/52

Mer om serialisering Serialiseringsprosessen er sekvensiell. Dette medfører at objektene blir deserialiert i den rekkefølgen de ble serialisert. Serialiseringsmekanismen unngår duplisering av objekter (Program 12.7). JFPS 12: Filer og strømmer 12-44/52

Direkte filtilgang (Figur 12.13) Klassen RandomAccessFile tilbyr det som kalles direkte filtilgang (Tabell 12.2). Kan både lese og skrive fra en vilkårlig posisjon i en fil. seek(0) It's now or never seek(inneværende - f) -f bytes inneværende = getfilepointer() +f bytes seek(inneværende + f) Flaming Star Suspious Minds Angel Crying in the Chapel If I can dream Fever CC Rider Little Sister Are you lonesome tonight? Devil in desguise seek(length()-1) //siste byte JFPS 12: Filer og strømmer 12-45/52

Direkte filtilgang (Tabell 12.2) RandomAccessFile(String filnavn, String filmodus) throws FileNotFoundException long length() void setlength(long nylengde) long getfilepointer() void seek(long forskyvning) int skipbytes(int n) boolean readboolean() char readchar() double readdouble() float readfloat() int readint() long readlong() java.io.randomaccessfile Åpner fil med angitt filnavn. Henter antall bytes lagret i filen, det vil si lengden til filen. Lengden til filen Henter filpekerens inneværende posisjon. Flytter filpekeren til posisjon angitt ved verdien i parameteren forskyvning. Hopper n bytes fremover fra inneværende posisjon i filen. Klassen definerer metoder for innlesing av binære verdier. JFPS 12: Filer og strømmer 12-46/52

void writeboolean(boolean v) void writechar(int v) void writedouble(double v) void writefloat(float v) void writeint(int v) void writelong(long v) void writechars(string s) void close() java.io.randomaccessfile Klassen definerer metoder for skriving av binære verdier. Metoden skriver hvert tegn i strengen som to bytes til strømmen, det vil si som et Unicode-tegn. Metoden lukker filen. JFPS 12: Filer og strømmer 12-47/52

Bruk av klassen RandomAccessFile For å opprette direkte tilgang til en fil må filen åpnes ved å angi filnavnet under opprettelsen av et RandomAccessFile-objekt: try { RandomAccessFile raf = new RandomAccessFile(datafilnavn, "rw"); } catch (FileNotFoundException unntak) {... } Filmodusen "rw" tillater kombinering av både lese- og skriveoperasjoner på filen. Filmodusen "r" tillater bare lesing fra en eksisterende fil. Antall bytes i en fil kalles fillengde, som returneres av metoden length(). En filpeker (eng. file pointer) angir hvor i filen lesing eller skriving kan utføres. En markør mellom to etterfølgende bytes på filen angir hvor lesing og skriving kan skje. Posisjoneringen av filpekeren med metoden seek() gjøres alltid ved å spesifisere riktig antall bytes fra starten av filen. JFPS 12: Filer og strømmer 12-48/52

Å hoppe over fem tegn fra inneværende posisjon: raf.skipbytes(10); For å utvide en fil ved å supplere data på slutten, må filpekerverdien settes lik fillengden: raf.seek(raf.length()); // Sett filpekeren til slutten av filen. Klassen RandomAccessFile kan også brukes til å lese og skrive binære verdier. En RandomAccessFile-strøm må også lukkes etter bruk. JFPS 12: Filer og strømmer 12-49/52

Eksempel: Bruk av direkte filtilgang (Program 12.8) Klassen AnsattDirekte leser og skriver en ansatt-post i posisjon angitt ved filpekeren. Alle poster har samme faste lengde. Dette betyr at samme felt i alle poster må ha fast feltlengde. For verdier av predefinerte datatyper er det gitt hvor mange bytes de opptar i filen. For at et felt med en strengverdi skal ha fast lengde, kan vi bruke følgende løsning: Dersom strengen er mindre enn feltlengden, fylles feltet med et bestemt tegn. Dersom strengen er større enn feltlengden, blir strengen trunkert slik at den passer i feltet. final static char FYLLE_TEGN = \u0000 ; final static int FORNAVN_LENGDE = 20; // Unicode-tegn final static int ETTERNAVN_LENGDE = 30; // Unicode-tegn final static int TIMELØNN_LENGDE = 8; // bytes final static int KJØNN_LENGDE = 1; // byte final static int POST_LENGDE_BYTES = 2*FORNAVN_LENGDE + 2*ETTERNAVN_LENGDE + TIMELØNN_LENGDE + KJØNN_LENGDE; // bytes JFPS 12: Filer og strømmer 12-50/52

Metoden skrivfastlengdestreng() i linje (3) tar strenglengden i betraktning og sørger alltid for at hele feltet blir fylt: int antallfylletegn = MAKS_FELT_LENGDE - str.length(); for (int k = 0; k < antallfylletegn; k++) str += FYLLE_TEGN; raf.writechars(str.substring(0, MAKS_FELT_LENGDE)); Metoden lesfastlengdestreng() i linje (5) sørger for at et felt som inneholder en streng, blir lest riktig: String str = ""; int i; for (i = 0; i < MAKS_FELT_LENGDE; i++) { char tegn = raf.readchar(); if (tegn == FYLLE_TEGN) break; else str += tegn; } // Hopp over ev. sekvens av FYLLE_TEGN som opptar 2 bytes hver. raf.skipbytes(2*(maks_felt_lengde - i - 1)); return str; Konstruktøren i klassen PersonellRegisterDirekte oppretter et RandomAccess- File-objekt i en try-blokk i linje (6). Den nullstiller først innholdet i filen ved å sette fillengden til null. JFPS 12: Filer og strømmer 12-51/52

Metoden utvidmedenansattpåfil() definert i linje (7) i klassen PersonellRegisterDirekte utvider filen med en ny post: // Posisjoner filpeker etter siste post. raf.seek(raf.length()); // Skriv ansatt-info til filen. ansatt.skrivansattdata(raf); og oppdaterer opplysninger om antall poster lagret ved starten av filen: raf.seek(0); int antallregistrert = raf.readint(); raf.seek(0); raf.writeint(antallregistrert + 1); Metoden registrernytimelønnforansattpåfil() i linje (8) oppdaterer direkte inn i filen timelønnen til en ansatt som er angitt ved et ansatt-nummer: raf.seek(4 + ansattnummer*ansattdirekte.post_lengde_bytes); Etter at posten er lest til et AnsattDirekte-objekt, blir timelønnet justert. Deretter blir filpekeren spolt tilbake slik at tidligere feltverdier i posten kan overskrives: long inneværendeposisjon = raf.getfilepointer(); raf.seek(inneværendeposisjon - AnsattDirekte.POST_LENGDE_BYTES); enansatt.skrivansattdata(raf); JFPS 12: Filer og strømmer 12-52/52