Java: Textový vstup a výstup

Z MiS
(Rozdíly mezi verzemi)
Přejít na: navigace, hledání
(Rozšířeno, další příklady,...)
(Opraveno na použití try-with-resources.)
 
(Není zobrazeno 8 mezilehlých verzí od 1 uživatele.)
Řádka 6: Řádka 6:
 
* Více o výjimkách si přečtete zde: [[Výjimky v Javě]].
 
* Více o výjimkách si přečtete zde: [[Výjimky v Javě]].
 
* Pro tuto chvíli bude stačit, když víte, že práci se soubory je vždy potřeba obklopit kódem pro řešení chyb:
 
* Pro tuto chvíli bude stačit, když víte, že práci se soubory je vždy potřeba obklopit kódem pro řešení chyb:
  try { ''// Začni provádět kód, ve kterém může nastat chybový stav...''
+
  try (''otevření souboru'') { ''// Otevři soubor a začni tím blok, ve kterém může nastat chybový stav...''
 
     ''... zde následuje kód, pracující se souborem...''
 
     ''... zde následuje kód, pracující se souborem...''
 
  } catch (IOException ex) {  
 
  } catch (IOException ex) {  
Řádka 14: Řádka 14:
 
  }  
 
  }  
 
* Více viz [[Výjimky v Javě]].
 
* Více viz [[Výjimky v Javě]].
 
  
 
== Čtení ze souboru (<code>Scanner</code>) ==
 
== Čtení ze souboru (<code>Scanner</code>) ==
Řádka 21: Řádka 20:
 
* Pro čtení dat z&nbsp;textových souborů slouží v Javě třída <code>Scanner</code> z&nbsp;balíčku <code>java.util</code>.
 
* Pro čtení dat z&nbsp;textových souborů slouží v Javě třída <code>Scanner</code> z&nbsp;balíčku <code>java.util</code>.
 
* Slouží také pro čtení vstupu z&nbsp;klávesnice v&nbsp;konzolových aplikacích (aplikacích, které běží jen v&nbsp;příkazovém řádku).
 
* Slouží také pro čtení vstupu z&nbsp;klávesnice v&nbsp;konzolových aplikacích (aplikacích, které běží jen v&nbsp;příkazovém řádku).
* Zaveden od Javy 5 &mdash; nahrazuje <tt>BufferedReader</tt>.
+
* Zaveden od Javy 5 &mdash; lze použít pro převod hodnot z&nbsp;textového zápisu jako „zapouzdření“ třídy <tt>BufferedReader</tt>.
 
* Výhodou je, že umí automaticky načíst čísla (včetně desetinných) z&nbsp;jejich textové podoby.
 
* Výhodou je, že umí automaticky načíst čísla (včetně desetinných) z&nbsp;jejich textové podoby.
  
Řádka 31: Řádka 30:
  
 
; Konstruktory třídy <code>Scanner</code>:
 
; Konstruktory třídy <code>Scanner</code>:
 +
* <code>Scanner(BufferedReader otevrenyVstupniStream, String encoding)</code>
 
* <code>Scanner(File vstupniSoubor, String encoding)</code>
 
* <code>Scanner(File vstupniSoubor, String encoding)</code>
 
* <code>Scanner(Path vstupniSoubor, String encoding)</code>
 
* <code>Scanner(Path vstupniSoubor, String encoding)</code>
Řádka 56: Řádka 56:
 
; Získání dalšího bloku dat ze souboru
 
; Získání dalšího bloku dat ze souboru
 
* <code>next()</code>
 
* <code>next()</code>
** přečte celý řádek až po oddělovač (viz dále)
+
** přečte další položku souboru až po oddělovač (viz dále)
 
** při čtení z konzole (klávesnice) čeká na oddělovač
 
** při čtení z konzole (klávesnice) čeká na oddělovač
 
** bere jednotlivá slova, oddělená standardně bílým místem
 
** bere jednotlivá slova, oddělená standardně bílým místem
Řádka 63: Řádka 63:
 
** vrací objektový typ <tt>Double</tt>.
 
** vrací objektový typ <tt>Double</tt>.
 
** Desetinná čísla je třeba zadávat podle národního nastavení. Pro české nastavení tedy s desetinnou čárkou.
 
** Desetinná čísla je třeba zadávat podle národního nastavení. Pro české nastavení tedy s desetinnou čárkou.
 +
* <code>String nextLine()</code>
 +
** Načte jeden řádek textu až po konec řádku (neřeší nastavení oddělovače/delimiteru).
  
 
; Zjištění &mdash; je v&nbsp;souboru další blok dat?
 
; Zjištění &mdash; je v&nbsp;souboru další blok dat?
 
* <code>boolean hasNext()</code>
 
* <code>boolean hasNext()</code>
 
** je k dispozici další slovo?
 
** je k dispozici další slovo?
 +
* <code>boolean hasNextInt()</code>
 +
** následuje na vstupu celé číslo?
 
* <code>boolean hasNextDouble()</code>
 
* <code>boolean hasNextDouble()</code>
 
** následuje na vstupu desetinné číslo?
 
** následuje na vstupu desetinné číslo?
 +
* <code>boolean hasNextLine()</code>
 +
** je na vstupu další celý řádek?
  
 
; Další nastavení
 
; Další nastavení
Řádka 88: Řádka 94:
 
  ''...''
 
  ''...''
 
   
 
   
  try {
+
  String nazev = "vstup.txt";
    String nazev = "vstup.txt";
+
Scanner sc = null;
    Scanner sc = new Scanner(new File(nazev), "windows-1250");
+
try (Scanner sc = new Scanner(new BufferedReader(new File(nazev)), "windows-1250");) {
     int cislo = sc.nextInt()
+
     int cislo = sc.nextInt();
     sc.close();
+
     ''// ...''
  } catch (FileNotFoundException ex) { System.err.println("Nenalezen soubor: "+nazev+"!"; }
+
  } catch (FileNotFoundException ex) { System.err.println("Nenalezen soubor: "+nazev+"!");
  } catch (IOException ex) { System.err.println("Chyba čtení ze souboru "+nazev+": "+ex.getLocalizedMessage()); }
+
  } catch (IOException ex) { System.err.println("Chyba čtení ze souboru "+nazev+": "+ex.getLocalizedMessage());
  } catch (InputMismatchException ex) { System.err.println("Nesprávný formát dat v souboru "+nazev+": "+ex.getLocalizedMessage()); }
+
  } catch (InputMismatchException ex) { System.err.println("Nesprávný formát dat v souboru "+nazev+": "+ex.getLocalizedMessage());  
   
+
  }
''...''
+
 
+
<div class="Poznamka">
 +
Pokud zadáte pouze název souboru, hledá se soubor v&nbsp;aktuální složce. Která složka je aktuální:
 +
* Při spuštění z&nbsp;vývojového prostředí (Greenfoot, NetBeans,...) je to obvykle složka, ve které jsou umístěny zdrojové soubory (na stejné úrovni jako složka <tt>src</tt>).
 +
* Při spuštění samostatného souboru .JAR je to složka, ve které je soubor .JAR umístěn.
 +
 
 +
Můžete zadat i celou cestu, například pro Windows: <code>String nazev = "C:\\temp\\data.txt";</code>, ale pak výsledný program funguje jen na vašem počítači a na počítačích, které mají soubor přesně ve stejném umístění.
 +
</div>
 +
 
  
 
=== Příklad: Načtení všech čísel ze souboru ===  
 
=== Příklad: Načtení všech čísel ze souboru ===  
Řádka 106: Řádka 119:
 
  ''...''
 
  ''...''
 
   
 
   
  try {
+
  String nazev = "vstup.txt";
    String nazev = "vstup.txt";
+
try (Scanner sc = new Scanner(new BufferedReader(new File(nazev))), "windows-1250");
    Scanner sc = new Scanner(new File(nazev), "windows-1250");
+
 
     while (sc.hasNextInt()) {
 
     while (sc.hasNextInt()) {
 
         int cislo = sc.nextInt();
 
         int cislo = sc.nextInt();
 
         System.out.println(cislo);
 
         System.out.println(cislo);
 
     }
 
     }
    sc.close();
+
  } catch (FileNotFoundException ex) { System.err.println("Nenalezen soubor: "+nazev+"!");  
  } catch (FileNotFoundException ex) { System.err.println("Nenalezen soubor: "+nazev+"!"; }
+
  } catch (IOException ex) { System.err.println("Chyba čtení ze souboru "+nazev+": "+ex.getLocalizedMessage());  
  } catch (IOException ex) { System.err.println("Chyba čtení ze souboru "+nazev+": "+ex.getLocalizedMessage()); }
+
  } catch (InputMismatchException ex) { System.err.println("Nesprávný formát dat v souboru "+nazev+": "+ex.getLocalizedMessage());  
  } catch (InputMismatchException ex) { System.err.println("Nesprávný formát dat v souboru "+nazev+": "+ex.getLocalizedMessage()); }  
+
}
 
   
 
   
 
  ''...''
 
  ''...''
Řádka 134: Řádka 146:
 
  Scanner sc = new Scanner(System.in);
 
  Scanner sc = new Scanner(System.in);
 
  int i = sc.nextInt();
 
  int i = sc.nextInt();
  vstup=input.next();
+
  String text = input.next();
 
  sc.close();
 
  sc.close();
  
Řádka 140: Řádka 152:
  
  
== Zápis do souboru (<tt>BufferedWriter</tt>) ==
+
== Zápis do souboru (<tt>PrintWriter</tt>) ==
  
Třída <code>BufferedWriter</code> slouží pro zápis dat do textového výstupu.
+
Třída <code>PrintWriter</code> slouží pro zápis dat do textového výstupu.
  
; Princip:
+
; Využívá efektivnější zápis pomocí vyrovnávací paměti (bufferu):
*Ukládá zapisovaná data do paměťového bufferu.
+
* Ukládá zapisovaná data do paměťového bufferu.
*Teprve při dosažení limitu data naráz zapíše.
+
* Teprve při dosažení limitu data naráz zapíše.
*Je to efektivnější, než psát jednotlivé malé texty hned!
+
* Je to efektivnější, než zapisovat jednotlivé malé kousky textu hned!
 
+
; Otevření standardního výstupu
+
BufferedWriter out = new BufferedWriter(new OutputStreamWriter(System.out));
+
Tuto variantu používáme tehdy, kdy chceme společnou metodu pro psaní do souboru i na standardní výstup.
+
Pro standardní výstup jinak samozřejmě stačí:
+
System.out.println(...);
+
  
 
; Otevření souboru
 
; Otevření souboru
  Charset kodovani = Charset.forName("utf-8");
+
  String kodovani = "windows-1250";
  Path cesta = Paths.get("vystup.txt");
+
  String nazev = "vystup.txt";
  BufferedWriter out = Files.newBufferedWriter(cesta, kodovani);
+
  PrintWriter out = new PrintWriter(new File(nazev), kodovani);
Související třídy:
+
 
* <tt>Charset</tt>
+
<div class="Poznamka">
** informace o kódování souboru
+
Pokud zadáte pouze název souboru, hledá se soubor v aktuální složce.
* <tt>Files</tt>, <tt>Paths</tt>
+
 
** práce s cestami a soubory
+
Která složka je aktuální:
** umí i otevřít <tt>BufferedWriter</tt>.
+
* Při spuštění z&nbsp;vývojového prostředí (Greenfoot, NetBeans,...) je to obvykle složka, ve které jsou umístěny zdrojové soubory.
 +
* Při spuštění samostatného souboru .JAR je to složka, ve které je soubor .JAR umístěn.
 +
 
 +
Můžete zadat i celou cestu, například pro Windows: <code>String nazev = "C:\\temp\\data.txt";</code>.
 +
</div>
  
 
; Metody
 
; Metody
* <code>write(String text)</code>
+
* <code>println(String text)</code>
* <code>newLine()</code>
+
** Přidá řádek do výstupního souboru.
** Přidá do výstupu konec řádku. Respektuje národní nastavení a platformové závislosti.
+
 
* <code>flush()</code>
 
* <code>flush()</code>
 
** Provede fyzický zápis stávajícího obsahu bufferu na výstup.
 
** Provede fyzický zápis stávajícího obsahu bufferu na výstup.
Řádka 176: Řádka 185:
 
** Provede i <code>flush()</code>.
 
** Provede i <code>flush()</code>.
  
Charset kodovani = Charset.forName("utf-8");
+
=== Příklad &mdash; Zápis věty do souboru ===
  Path cesta = Paths.get("vystup.txt");
+
  import java.io.File;
  BufferedWriter out = Files.newBufferedWriter(cesta, kodovani);
+
  import java.io.PrintWriter;
out.write("blabla");
+
  import java.io.FileNotFoundException;
  out.newLine();
+
  import java.io.UnsupportedEncodingException;
  out.close();
+
  
 +
''...''
  
 +
String kodovani = "windows-1250";
 +
String nazevSouboru = "vystup.txt";
 +
try (PrintWriter out = new PrintWriter(new File(nazevSouboru), kodovani))
 +
    out.println("Hello world!");
 +
    out.close();
 +
} catch (FileNotFoundException ex) { System.err.println("Soubor "+nazevSouboru+" nenalezen: "+ex.getLocalizedMessage());
 +
} catch (UnsupportedEncodingException ex) { System.err.println("Neznámé kódování "+kodovani+": "+ex.getLocalizedMessage());
 +
}
  
 +
 +
<div class="Poznamka">
 +
; Můžete použít i&nbsp;novější knihovnu <code>java.nio</code> a&nbsp;třídu <code>BufferedWriter</code>
 +
Museli byste si ale sami vyřešit textový zápis celých a&nbsp;desetinných čísel:
 +
Charset kodovani = Charset.forName("windows-1250");
 +
Path cesta = Paths.get("vystup.txt");
 +
BufferedWriter out = Files.newBufferedWriter(cesta, kodovani);
 +
Související třídy:
 +
* <tt>Charset</tt>
 +
** informace o kódování souboru
 +
* <tt>Files</tt>, <tt>Paths</tt>
 +
** práce s cestami a soubory.
 +
</div>
  
 
== Standardní vstup a výstup ==
 
== Standardní vstup a výstup ==
Řádka 202: Řádka 232:
 
  System.in.available() > 0
 
  System.in.available() > 0
  
 +
; System.err
 +
* Chybový výstup. Na obrazovce se vypíše stejně jako u&nbsp;<code>System.out</code>, ale můžeme ho přesměrovat (viz práce se vstupy a&nbsp;výstupy ve 3.&nbsp;ročínku (Operační systémy):
 +
System.err.println("Nastala chyba zápisu do souboru!");
  
 
== Související stránky ==
 
== Související stránky ==

Aktuální verze z 7. 11. 2022, 10:07


Obsah

Ošetření chyb, výjimky

try (otevření souboru) { // Otevři soubor a začni tím blok, ve kterém může nastat chybový stav...
    ... zde následuje kód, pracující se souborem...
} catch (IOException ex) { 
    // Co se má stát, pokud nastane chyba při práci se souborem...
} catch (FileNotFoundException ex) { 
    // Co se má stát, když zadaný soubor vůbec neexistuje
} 

Čtení ze souboru (Scanner)

Třída Scanner

Třídu Scanner nepoužívejte v programech, které pracují s vlákny!

Otevření souboru

Konstruktory třídy Scanner

Kódování národních znaků:


Pokud nenastavíte správné kódování českých znaků, třída Scanner se při ladění v prostředí NetBeans 8.0.2 (JDK 7u71) může chovat k souboru, jako by byl prázdný.

Pokud máte podobné problémy, zkuste:

a) nastavte správné kódování češtiny v konstruktoru Scanneru či zkonvertujte soubor na správné kódování (třeba pomocí Notepadu++ či jiného editoru, který umí nastavit kódování).

b) projekt přeložte (Clean & Build) a spusťte přímo výsledný JAR ze složky dist v adresáři projektu. (viz Export balíčku JAR).

Metody třídy Scanner

Získání dalšího bloku dat ze souboru
Zjištění — je v souboru další blok dat?
Další nastavení
out.useDelimiter("\\s*[;,\n\r\t]\\s*")
Ukončení práce se souborem

Příklad: Načtení jednoho čísla ze souboru

import java.util.Scanner;
import java.io.File;

...

String nazev = "vstup.txt";
Scanner sc = null;
try (Scanner sc = new Scanner(new BufferedReader(new File(nazev)), "windows-1250");) {
    int cislo = sc.nextInt();
    // ...
} catch (FileNotFoundException ex) { System.err.println("Nenalezen soubor: "+nazev+"!");
} catch (IOException ex) { System.err.println("Chyba čtení ze souboru "+nazev+": "+ex.getLocalizedMessage());
} catch (InputMismatchException ex) { System.err.println("Nesprávný formát dat v souboru "+nazev+": "+ex.getLocalizedMessage()); 
}

Pokud zadáte pouze název souboru, hledá se soubor v aktuální složce. Která složka je aktuální:

  • Při spuštění z vývojového prostředí (Greenfoot, NetBeans,...) je to obvykle složka, ve které jsou umístěny zdrojové soubory (na stejné úrovni jako složka src).
  • Při spuštění samostatného souboru .JAR je to složka, ve které je soubor .JAR umístěn.

Můžete zadat i celou cestu, například pro Windows: String nazev = "C:\\temp\\data.txt";, ale pak výsledný program funguje jen na vašem počítači a na počítačích, které mají soubor přesně ve stejném umístění.


Příklad: Načtení všech čísel ze souboru

import java.util.Scanner;
import java.io.File;

...

String nazev = "vstup.txt";
try (Scanner sc = new Scanner(new BufferedReader(new File(nazev))), "windows-1250");
    while (sc.hasNextInt()) {
        int cislo = sc.nextInt();
        System.out.println(cislo);
    }
} catch (FileNotFoundException ex) { System.err.println("Nenalezen soubor: "+nazev+"!"); 
} catch (IOException ex) { System.err.println("Chyba čtení ze souboru "+nazev+": "+ex.getLocalizedMessage()); 
} catch (InputMismatchException ex) { System.err.println("Nesprávný formát dat v souboru "+nazev+": "+ex.getLocalizedMessage()); 
}

...


Příklad: Načtení všech řádků textu ze souboru

Scanner sc = new Scanner(new File("myRows"), "windows-1250");
while (sc.hasNextLine()) {
    ...
    String radek = sc.nextLine();
    ...
}
sc.close();

Příklad: Načtení jednoho čísla z klávesnice

Scanner sc = new Scanner(System.in);
int i = sc.nextInt();
String text = input.next();
sc.close();



Zápis do souboru (PrintWriter)

Třída PrintWriter slouží pro zápis dat do textového výstupu.

Využívá efektivnější zápis pomocí vyrovnávací paměti (bufferu)
Otevření souboru
String kodovani = "windows-1250";
String nazev = "vystup.txt";
PrintWriter out = new PrintWriter(new File(nazev), kodovani);

Pokud zadáte pouze název souboru, hledá se soubor v aktuální složce.

Která složka je aktuální:

  • Při spuštění z vývojového prostředí (Greenfoot, NetBeans,...) je to obvykle složka, ve které jsou umístěny zdrojové soubory.
  • Při spuštění samostatného souboru .JAR je to složka, ve které je soubor .JAR umístěn.

Můžete zadat i celou cestu, například pro Windows: String nazev = "C:\\temp\\data.txt";.

Metody

Příklad — Zápis věty do souboru

import java.io.File;
import java.io.PrintWriter;
import java.io.FileNotFoundException;
import java.io.UnsupportedEncodingException;
...
String kodovani = "windows-1250";
String nazevSouboru = "vystup.txt";
try (PrintWriter out = new PrintWriter(new File(nazevSouboru), kodovani))
    out.println("Hello world!");
    out.close();
} catch (FileNotFoundException ex) { System.err.println("Soubor "+nazevSouboru+" nenalezen: "+ex.getLocalizedMessage());
} catch (UnsupportedEncodingException ex) { System.err.println("Neznámé kódování "+kodovani+": "+ex.getLocalizedMessage());
}


Můžete použít i novější knihovnu java.nio a třídu BufferedWriter

Museli byste si ale sami vyřešit textový zápis celých a desetinných čísel:

Charset kodovani = Charset.forName("windows-1250");
Path cesta = Paths.get("vystup.txt");
BufferedWriter out = Files.newBufferedWriter(cesta, kodovani);

Související třídy:

  • Charset
    • informace o kódování souboru
  • Files, Paths
    • práce s cestami a soubory.

Standardní vstup a výstup

System.out
System.out.print("Ahoj!");
System.out.println("Vypíše řádek na obrazovku!");
System.in
System.in.read();
System.in.available() > 0
System.err
System.err.println("Nastala chyba zápisu do souboru!");

Související stránky


Zdroje

  1. JavaPractices.com → Reading and writing text files
  2. Java Tutorial → Scanner
Osobní nástroje
Jmenné prostory
Varianty
Akce
Výuka
Navigace
Nástroje