Válí se ti projekty v šuplíku? Dostaň je mezi lidi a získej cool tričko a body na profi IT kurzy v soutěži ITnetwork summer 2017!
Přidej si svou IT školu do profilu a najdi spolužáky zde na síti :)

3. díl - Práce s textovými soubory v Javě

Java Práce se soubory Práce s textovými soubory v Javě

Unicorn College ONEbit hosting Tento obsah je dostupný zdarma v rámci projektu IT lidem. Vydávání, hosting a aktualizace umožňují jeho sponzoři.

V minulém dílu seriálu tutoriálů pro Javu jsme si ukázali, jak fungují přístupová práva v systémech Windows. Nejjednodušší cestou, jak uložit data aplikace na pevný disk, je využít textové soubory. Se soubory s příponou .txt jste se jistě všichni již setkali. Text je v nich uložen jednoduše na jednotlivých řádcích. K oddělení řádků se využívá speciálních znaků, které jsou bohužel specifické pro každý operační systém. Toto však za nás naštěstí vyřeší Java.

Pozn.: V různých materiálech se můžete setkat s použitím různých tříd a různých způsobů k zápisu do souborů. Budu se zde snažit ukazovat způsoby, které jsou nejjednodušší a nejnovější.

Zápis textu do nového souboru

Nejprve si pojďme vytvořit nový textový soubor a něco do něj zapsat. Vytvořte si nový projekt s názvem TextoveSoubory. K zapisování do textových souborů nám Java poskytuje třídu BufferedWriter. Buffered znamená, že zapisovač využívá vyrovnávací paměť, aby urychlil zápis na disk.

Vytvořme blok try-with-resources a založme v něm novou instanci BufferedWriteru. Jak již víme z předchozích dílů, try-with-resources se nám automaticky postará o zavření souboru po dokončení zápisu/čtení. Do konstruktoru vložíme instanci FileWriteru, který BufferedWriter obaluje.

try (BufferedWriter bw = new BufferedWriter(new FileWriter("soubor.txt")))
{
}
catch (Exception e)
{
        System.err.println("Do souboru se nepovedlo zapsat.");
}

Za zmínku stojí použití System.err v bloku catch, což je chybový kanál pro výstup. V případě, že se zápis opravdu nepovede, uvidíte tento text v konzoli červenou barvou.

Náš BufferedWriter je nyní nasměrovaný na správný soubor. Nový řádek zapíšeme pomocí metody write(). Odřádkování v souboru docílíme metodou newLine(). Po dokončení zápisu musíme zavolat metodu flush(), která se stará o vyprázdnění bufferu. S tím se zde nebudeme zatěžovat, postačí nám vědět, že námi zapsané řádky mohou zůstat chvíli ve vyrovnávací paměti a my pomocí flush() vynutíme jejich zápis.

Kód se nám tedy rozrostl a může vypadat např. takto:

try (BufferedWriter bw = new BufferedWriter(new FileWriter("soubor.txt")))
{
        bw.write("První řádek");
        bw.newLine();
        bw.write("Tento text je na druhém řádku");
        bw.newLine();
        bw.write("A do třetice.");
        bw.newLine();
        bw.flush();
}
catch (Exception e)
{
        System.err.println("Do souboru se nepovedlo zapsat.");
}

Po spuštění se vytvoří soubor.txt ve složce s naším projektem. Jak se správně v Javě vypořádat s cestou k souboru a právy jsme si již ukazovali a zde to pro jednoduchost zanedbáme. Vidíme, že soubor existuje a opravdu obsahuje náš text:

Zápis textových souborů v Javě

Připsání textu do existujícího souboru

Pokud soubor neexistuje, kód výše ho vytvoří. Pokud existuje, bude přepsán. Toto chování můžeme změnit pomocí 2. parametru konstruktoru objektu FileWriter. Pokud ho nastavíme na true, provede se tzv. append (připsání). Takto do existujícího souboru připíšeme nový řádek:

try (BufferedWriter bw = new BufferedWriter(new FileWriter("soubor.txt", true)))
{
        bw.write("Připsaný řádek");
        bw.newLine();
        bw.flush();
}
catch (Exception e)
{
        System.err.println("Do souboru se nepovedlo zapsat.");
}

Pokud soubor neexistuje, bude vytvořen. Pokud existuje, bude do něj připsáno a jeho původní obsah zůstane.

Čtení existujícího souboru

Zbývá nám již jen umět soubor načíst. Není to o nic složitější, než zápis a opět k tomu máme v Javě připravenou třídu, konkrétně BufferedReader. Použití je obdobné, namísto metody write() použijeme readLine(), která vrací řádek textu ze souboru a zároveň se přesune na řádek následující. Budeme ji tedy volat ve while cyklu. Podmínka pro ošetření vyjetí ze souboru je možná krkolomnější, kontrolujeme, zda proběhlo přiřazení nové řádky do proměnné.

Kód k výpisu obsahu souboru do konzole by vypadal takto:

try (BufferedReader br = new BufferedReader(new FileReader("soubor.txt")))
{
        String s;
        while ((s = br.readLine()) != null)
        {
                System.out.println(s);
        }
}
catch (Exception e)
{
        System.err.println("Chyba při četení ze souboru.");
}

Kód celého našeho programu vypadá nyní asi takto:

// zápis do souboru
try (BufferedWriter bw = new BufferedWriter(new FileWriter("soubor.txt")))
{
        bw.write("První řádek");
        bw.newLine();
        bw.write("Tento text je na druhém řádku");
        bw.newLine();
        bw.write("A do třetice.");
        bw.newLine();
        bw.flush();
        System.out.println("Do souboru bylo zapsáno");
}
catch (Exception e)
{
        System.err.println("Do souboru se nepovedlo zapsat.");
}

// připsání textu do existujícího souboru
try (BufferedWriter bw = new BufferedWriter(new FileWriter("soubor.txt", true)))
{
        bw.write("Připsaný řádek");
        bw.newLine();
        bw.flush();
        System.out.println("Do souboru bylo připsáno");
}
catch (Exception e)
{
        System.err.println("Do souboru se nepovedlo zapsat.");
}

// výpis obsahu souboru
System.out.println("Vypisuji celý soubor:");
try (BufferedReader br = new BufferedReader(new FileReader("soubor.txt")))
{
        String s;
        while ((s = br.readLine()) != null)
        {
                System.out.println(s);
        }
}
catch (Exception e)
{
        System.err.println("Chyba při četení ze souboru.");
}

A výsledek:

Zápis a čtení textových souborů v Javě

Opominuli jsme kontrolu práv. Do souborů také budeme chtít většinou ukládat spíše objekty, než řádky textu. To vše si ukážeme dále.


 

Stáhnout

Staženo 615x (17.16 kB)
Aplikace je včetně zdrojových kódů v jazyce java

 

 

Článek pro vás napsal David Čápka
Avatar
Jak se ti líbí článek?
5 hlasů
Autor pracuje jako softwarový architekt a pedagog na projektu ITnetwork.cz (a jeho zahraničních verzích). Velmi si váží svobody podnikání v naší zemi a věří, že když se člověk neštítí práce, tak dokáže úplně cokoli.
Unicorn College Autor se informační technologie naučil na Unicorn College - prestižní soukromé vysoké škole IT a ekonomie.
Miniatura
Předchozí článek
Úvod do práce se soubory v Javě
Miniatura
Všechny články v sekci
Práce se soubory v Javě
Miniatura
Následující článek
Úvod do formátu XML souborů v Javě
Aktivity (1)

 

 

Komentáře
Zobrazit starší komentáře (8)

Avatar
vit-siler
Člen
Avatar
vit-siler:10.8.2014 22:34

Jak se to tedy da napsat pomoci "try-with-resources" bez metody flush()??

 
Odpovědět 10.8.2014 22:34
Avatar
andrej200
Člen
Avatar
andrej200:7.5.2015 12:17

A co když chci přistupovat k souboru v adresáři např: C:\Games ?

Odpovědět 7.5.2015 12:17
Boj sa tých, ktorí sú ticho. Oni sú totiž tí jediní, ktorí skutočne myslia.
Avatar
Odpovídá na andrej200
Michal Žůrek (misaz):7.5.2015 21:33

pokud máš dostatek oprávnění tak v tom není problém.

Odpovědět 7.5.2015 21:33
Nesnáším {}, proto se jim vyhýbám.
Avatar
ronep
Člen
Avatar
ronep:21.6.2015 13:09

a ako nacitat riadok ktori chcem ja ?

 
Odpovědět 21.6.2015 13:09
Avatar
matrin
Člen
Avatar
matrin:26.12.2015 22:17

Potřebuji pracovat se soubory, tzn. v programu potřebuji z nějakého souboru načíst text a pak s ním i pracovat a do stejného souboru ukládat nějaké data. V jedné metodě mam, aby se zadal od uzivatele nazev souboru do promenne, ze ktere se pak cte. Z metody se vrací jit načteny text pod Stringem. Potřebuji však dále pracovat a přenášet do ostatních metod promennou, ve které je název souboru, aby se to dále ukládala data do zadaného souboru. Jste mi nekdo ochoten poradit, jestli jste z textu nějak pochopili muj problém? Díky

Takto vypadá ta metoda pro čtení ze souboru:

static String nactiText(String s) throws IOException { //funkce pro nacteni textu ze souboru
Scanner sc = new Scanner(System.in);
String soubor;
String nactenyTxt;

System.out.prin­tln(s);
soubor = sc.nextLine();

try (BufferedReader in = new BufferedReader(new FileReader(sou­bor))) {

nactenyTxt = in.readLine();
return nactenyTxt;

} catch (Exception e) {
System.err.prin­tln("Chyba při četení ze souboru.");
return "";
}
}

 
Odpovědět 26.12.2015 22:17
Avatar
pocitac770
Redaktor
Avatar
Odpovídá na matrin
pocitac770:26.12.2015 23:17

Ten text z proměnné "soubor" si můžeš překopírovat do nějaké globální statické proměnné. Pro pozdějsí užití by bylo možná ještě lepší si oddělit načítání souboru zvlášť do jiné metody a tu pak volat z mainu... Ale to je jen tip :)

Editováno 26.12.2015 23:18
 
Odpovědět 26.12.2015 23:17
Avatar
matrin
Člen
Avatar
Odpovídá na pocitac770
matrin:27.12.2015 21:52

Díky, už jsem to vyřešil. Přišel jsem ještě na další problém...pokud chci načíst text ze souboru, načte to pouze slova do prvního odstavce, tzn. dokud v textu není další řádek, když zmáčknu enter. Potřebuju aby mi to načetlo cely text, lze to nejak? Díky

 
Odpovědět 27.12.2015 21:52
Avatar
pocitac770
Redaktor
Avatar
Odpovídá na matrin
pocitac770:27.12.2015 22:26
String text;
while((text = in.readLine()) != null)
        nactenyTxt += text;

Něco takového (místo řádky "nactenyTxt = in.readLine();" )? Je to vlastně na stejném principu jako první kód pod "Čtení existujícího souboru" ;)

Editováno 27.12.2015 22:28
 
Odpovědět 27.12.2015 22:26
Avatar
koumes3
Člen
Avatar
koumes3:29. ledna 19:27

Ahoj, chci se zeptat, zda už se někdo setkal s následujícím problémem.
Při načítání textu ze souboru, kde jsou v textu obsažené i znaky pro odřádkování a tabulátor nebo escapování uvozovek, se do textu uloží(zdvojí se) zpětné lomítka. Dočetl jsem se, že to je způsobeno tím, že se čtou znaky po jednom a tudíž narazí-li se na \ tak se automaticky escapuje a tím vznikne to dvojité \\. Vypadá to pak např. \\n \\t nebo \\". To vcelku chápu.Na co, ale nemůžu přijít ani se dočíst je, jak to zdvojení odstranit, aby pak se text zobrazil jak má. U tabulátoru a odřádkování se dá použít např. text.replace("\\t", "\t"); což ovšem u uvozovek neprojde. V kódu to hlásí chybu. Pokud by měl někdo nějaké elegantní řešení budu rád, když se se mnou podělí o zkušenost.
Díky

 
Odpovědět 29. ledna 19:27
Avatar
koumes3
Člen
Avatar
koumes3:29. ledna 19:31

Ještě k předchozímu dotazu. Program jsem zkoušel v NetBeans IDE.

 
Odpovědět 29. ledna 19:31
Děláme co je v našich silách, aby byly zdejší diskuze co nejkvalitnější. Proto do nich také mohou přispívat pouze registrovaní členové. Pro zapojení do diskuze se přihlas. Pokud ještě nemáš účet, zaregistruj se, je to zdarma.

Zobrazeno 10 zpráv z 18. Zobrazit vše