Výjimky v Javě

Z MiS
(Rozdíly mezi verzemi)
Přejít na: navigace, hledání
(Přidán příklad.)
m (Opravy kódu.)
Řádka 65: Řádka 65:
 
** šířku pozemku (celé číslo)
 
** šířku pozemku (celé číslo)
 
** délku pozemku (celé číslo)
 
** délku pozemku (celé číslo)
** popis typu pozemku (text)
+
** popis pozemku (text)
 
* Položky jsou na řádku odděleny středníkem.
 
* Položky jsou na řádku odděleny středníkem.
 
* Budeme postupně načítat jednotlivé řádky souboru a vypisovat je na obrazovku.
 
* Budeme postupně načítat jednotlivé řádky souboru a vypisovat je na obrazovku.
Řádka 72: Řádka 72:
  
 
  import java.util.Scanner;
 
  import java.util.Scanner;
  import java.io.File;
+
  import java.io.*;
 +
import java.util.InputMismatchException;
  
 
  ...
 
  ...
  
 +
String nazev = "vstup.txt";
 +
Scanner sc = null;
 
  '''try''' {
 
  '''try''' {
     String nazev = "vstup.txt";
+
     sc = new Scanner(new File(nazev), "windows-1250");
    Scanner sc = new Scanner(new File(nazev), "windows-1250");
+
 
     while (sc.hasNextLine()) {
 
     while (sc.hasNextLine()) {
 
         int sirka = sc.nextInt();
 
         int sirka = sc.nextInt();
 
         int delka = sc.nextInt();
 
         int delka = sc.nextInt();
         String nazev = sc.next();
+
         String popis = sc.next();
 
         sc.nextLine();
 
         sc.nextLine();
         System.out.println("Pozemek: "+nazev+" má šířku: "+sirka+" m a délku: "+delka+"\n");
+
         System.out.println("Pozemek: "+popis+" má šířku: "+sirka+" m a délku: "+delka+" m.");
         System.out.println("Plocha pozemku je: "+(sirka*delka)+" m^2");
+
         System.out.println("Plocha pozemku je: "+(sirka*delka)+" m^2.");
 
         System.out.println("---------------");
 
         System.out.println("---------------");
 
     }
 
     }
  } '''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());
 
  } '''finally''' {
 
  } '''finally''' {
     '''try''' {
+
     sc.close();
        sc.close();
+
    } '''catch''' (IOException ex) {
+
        System.err.println("Došlo k chybě při zavírání souboru "+nazev+": "+ex.getLocalizedMessage());
+
    }
+
 
  }
 
  }
  

Verze z 14. 4. 2020, 13:36


Obsah

Co jsou výjimky?

Ještě jednou pro ujasnění:

  • Neřešíme teď situaci, kdy je aplikace špatně naprogramovaná!
  • Aplikace je správně napsaná, ale problém nastane v něčem, co programátor nemůže dopředu ovlivnit.
  • Například výpadek sítě při komunikaci přes síť. Takové chyby lze předvídat a můžeme do aplikace zabudovat vhodnou reakci na takovou chybu. Ale nelze zajistit, aby taková chyba nenastala.


Jak výjimky fungují?

  1. Označíme blok kódu, kde chyba může nastat:
    try { ... }
  2. Následně zapíšeme kód bez toho, abychom museli neustále testovat chyby
    • Kód tedy může tedy zůstat přehledný a stručný, bez spousty podmínek.
    • Pokud chyba nastane, zbytek bloku kódu se přeskočí a ihned se provede reakce na chybu (viz další krok).
  3. Poté popíšeme, jak zareagovat na jednotlivé typy chyb (jak zareagovat na výjimky):
    catch (TypVýjimky názevVýjimky) { ... jak zareagovat ... }
    • Počet typů chyb, na které reagujeme, není omezený.
  4. A nakonec můžeme (ale nemusíme) uvést kód, který se má provést bez ohledu na to, jestli nastala chyba:
    finally { ... Co se má stát vždy na konci bloku ... }


Šablona kódu ošetření výjimek

try { // Začni provádět kód, ve kterém může nastat chybový stav...

    ... zde následuje kód, pracující se souborem...

} catch (TypVýjimky1 ex) { 

    // Co se má stát, pokud nastane chyba TypVýjimky1...

} catch (TypVýjimky2 ex) { 

    // Co se má stát, pokud nastane chyba TypVýjimky2...

    // Takto můžeme popsat reakci na různé typy výjimek.

} finally {

    // Co se má stát každopádně po skončení bloku...
    // Provede se i tehdy, kdy nastane neočekávaná chyba.

    // Část finally není povinná, může se vynechat...

}


Příklad — práce se souborem

import java.util.Scanner;
import java.io.*;
import java.util.InputMismatchException;
...
String nazev = "vstup.txt";
Scanner sc = null;
try {
    sc = new Scanner(new File(nazev), "windows-1250");
    while (sc.hasNextLine()) {
        int sirka = sc.nextInt();
        int delka = sc.nextInt();
        String popis = sc.next();
        sc.nextLine();
        System.out.println("Pozemek: "+popis+" má šířku: "+sirka+" m a délku: "+delka+" m.");
        System.out.println("Plocha pozemku je: "+(sirka*delka)+" m^2.");
        System.out.println("---------------");
    }
} 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());
} finally {
    sc.close();
}
...


Delegování chyb   throws

Například
  • Píšeme knihovní funkci, která bude sloužit pro načítání XML souborů v různých aplikacích.
  • Přitom ale nevíme, jaká aplikace to bude:
    • některé aplikace mohou být textové,
    • jiné mohou mít grafické uživatelské rozhraní,
    • a některé aplikace mohou být třeba napsány v Greenfootu.
  • Každá aplikace tedy bude chtít zareagovat na chybu jinak:
    • v textovém režimu vypsáním hlášení pomocí System.err.println(),
    • v grafickém režimu pomocí vyskakovacího okna
    • a v Greenfootu třeba tak, že nějakému aktérovi nastavíme text s popisem hlášení.
  • Chybu tedy nechceme ošetřovat hned, protože ještě nevíme jak to nejlépe provést.
private void nactiSoubor() throws IOException {
    ... kód, kde může nastat chyba IOException, na kterou ale v tomto místě neumíme správně zareagovat...
    ... další kód, který se v případě chyby přeskočí...
}
public void zpracuj() {
    try {
        ... nějaký kód... 
        nactiSoubor();
        ... nějaký kód - pokud v metodě nactiSoubor() nastane chyba, tento kód se přeskočí... 
    } catch (IOException ex) { ... jak reagovat na chybu... 
    }
}


Vyhození vlastní výjimky   throw

if (castka > limit) {
    int rozdil = castka - limit;
    throw new MyException("Překročen limit o: "+rozdil+" Kč");
}


Související stránky


Zdroje

Osobní nástroje
Jmenné prostory
Varianty
Akce
Výuka
Navigace
Nástroje