PO Wyjątki - ćwiczenia: Różnice pomiędzy wersjami
Linia 209: | Linia 209: | ||
} | } | ||
} '''catch''' (Exception w) {} | } '''catch''' (Exception w) {} | ||
} | |||
} | |||
</div> | |||
</div> | |||
== Zadanie 4 == | |||
W poniższym przykładzie w bloku '''try''' jest pięć instrukcji. Każda z nich może zgłosić taki sam wyjątek. Zaproponuj rozwiązanie, które w kodzie obsługi wyjątku pozwoli sprawdzić w której instrukcji on wystąpił. | |||
'''public''' '''class''' Zad4 { | |||
'''void''' możeZgłosićWyjątek() '''throws''' Exception { | |||
'''if''' ('''new''' java.util.Random().nextInt(5) == 0) | |||
'''throw''' '''new''' Exception(); | |||
} | |||
'''public''' '''static''' '''void''' main(String[] args) { | |||
Zad4 z = '''new''' Zad4(); | |||
'''try''' { | |||
z.możeZgłosićWyjątek(); | |||
z.możeZgłosićWyjątek(); | |||
z.możeZgłosićWyjątek(); | |||
z.możeZgłosićWyjątek(); | |||
z.możeZgłosićWyjątek(); | |||
} '''catch''' (Exception e) { | |||
//jakiś kod obsługi | |||
} | |||
} | |||
} | |||
<div class="mw-collapsible mw-made=collapsible mw-collapsed"> | |||
'''Rozwiązanie''' | |||
<div class="mw-collapsible-content" style="display:none"> | |||
'''public''' '''class''' Zad4 { | |||
'''void''' możeZgłosićWyjątek() '''throws''' Exception { | |||
'''if''' ('''new''' java.util.Random().nextInt(5) == 0) | |||
'''throw''' '''new''' Exception(); | |||
} | |||
'''public''' '''static''' '''void''' main(String[] args) { | |||
Zad4 z = '''new''' Zad4(); | |||
int i = 1; | |||
'''try''' { | |||
z.możeZgłosićWyjątek(); | |||
i++; | |||
z.możeZgłosićWyjątek(); | |||
i++; | |||
z.możeZgłosićWyjątek(); | |||
i++; | |||
z.możeZgłosićWyjątek(); | |||
i++; | |||
z.możeZgłosićWyjątek(); | |||
i++; | |||
} '''catch''' (Exception e) { | |||
System.out.println("Wyjątek wystąpił w instrukcji "+ i); | |||
//jakiś kod obsługi | |||
} | |||
'''if''' (i == 6) System.out.println("Wyjątek nie wystąpił"); | |||
} | } | ||
} | } |
Wersja z 11:20, 30 lip 2006
Zadanie 1
Napisz metodę, które jako parametr będzie przyjmowała napis i wypisywała na standardowe wyjście jego długość.
- Przekaż do tej metody null i zobacz jaki wyjątek został zgłoszony.
- Otocz wywołanie metody blokiem try-catch, przechwyć ten wyjątek i wypisz na standardowe wyjście ślad stosu wywołań z chwili zgłoszenia wyjątku.
- Bezpośrednio po wypisaniu jego śladu zgłoś obsługiwany wyjątek ponownie.
- Czy ślady stosu wypisane przez ciebie w bloku catch i przez maszynę wirtualną w chwili przerwania programu są takie same?
- Przed zgłoszeniem ponownie obsługiwanego wyjątku wykonaj na nim metodę fillInStackTrace().
- Zamiast zgłaszać ponownie obsługiwany wyjątek zgłoś nowy wyjątek klasy Exception. Zauważ, że wymaga to pewnej dodatkowej zmiany w kodzie.
- Dołącz obsługiwany wyjątek do nowo tworzonego wyjątku Exception jako przyczynę jego powstania.
Rozwiązanie (część 1)
Rozwiązanie (część 2)
Rozwiązanie (część 3)
Rozwiązanie (część 4)
Rozwiązanie (część 5)
Rozwiązanie (część 6)
Rozwiązanie (część 7)
Zadanie 2
Napisz trzy nowe wyjątki oraz metodę, która za każdym wywołanie będzie losowo zgłaszała jeden z nich.
- Otocz wywołanie tej metody instrukcją try-catch zawierającą po klauzuli catch dla każdego z wyjątków. Niech każda klauzula catch wypisuje który wyjątek złapała.
- Sprawdź co się stanie jeżeli zmienisz kod metody tak żeby przestała zgłaszać jeden z wyjątków, ale nadal będzie on wymieniony w klauzuli throws nagłówka metody. Czy metody wywołujące napisaną przez ciebie metodę, ale nie obsługujące tego wyjątku będą musiały nadal go wymieniać w swoich nagłówkach?
Rozwiązanie (część 1)
Rozwiązanie (część 2)
Zadanie 3
Napisz program sprawdzający, czy wyjątki zgłoszone w klauzuli catch mogą być obsłużone przez tą samą klauzulę lub jedną z pozostałych klauzul tej samej instrukcji try-catch.
Rozwiązanie
Zadanie 4
W poniższym przykładzie w bloku try jest pięć instrukcji. Każda z nich może zgłosić taki sam wyjątek. Zaproponuj rozwiązanie, które w kodzie obsługi wyjątku pozwoli sprawdzić w której instrukcji on wystąpił.
public class Zad4 { void możeZgłosićWyjątek() throws Exception { if (new java.util.Random().nextInt(5) == 0) throw new Exception(); } public static void main(String[] args) { Zad4 z = new Zad4(); try { z.możeZgłosićWyjątek(); z.możeZgłosićWyjątek(); z.możeZgłosićWyjątek(); z.możeZgłosićWyjątek(); z.możeZgłosićWyjątek(); } catch (Exception e) { //jakiś kod obsługi } } }
Rozwiązanie
Zadanie 4
Zmodyfikuj klasę ZwalnianieZasobów2, aby zwalnianie odbywało się w klauzuli finally. Sprawdź, czy rzeczywiście zasób jest zwalniany nawet jak blok try jest opuszczany przy pomocy instrukcji return, break i continue.
Rozwiązanie
Zadanie 5
Podczas rezerwowania i zwalniania zasobów zazwyczaj też może wystąpić wyjątek. Użyj pokazanej poniższej klasy Zasób3 i zmodyfikuj swoje rozwiązanie poprzedniego zadania, tak aby zawsze oba zasoby były zwalniane. Zadbaj żeby nie dochodziło do zaginięcia żadnych wyjątków. Możesz założyć, że wykonanie operacji zwolnij() na zamkniętym zasobie jest dozwolone.
class InnyMożliwyWyjątek extends Exception {} class JeszczeInnyMożliwyWyjątek extends Exception {} class Zasób3 { Zasób3(int i) { //... } void zarezerwuj() throws IOException {} void używaj() throws IOException {} void innaNiebezpiecznaOperacja() throws InnyMożliwyWyjątek, JeszczeInnyMożliwyWyjątek {} void zwolnij() throws IOException {} }
Rozwiązanie
Zadanie 6
Sprawdź, czy finally jest wykonywane pomimo wystąpienia wyjątku w bloku catch z tej samej instrukcji.
Rozwiązanie