Wyjątki

Autor podstrony: Krzysztof Zajączkowski

Stronę tą wyświetlono już: 3601 razy

Kiedy coś pójdzie nie tak...

No właśnie, gdy coś pójdzie nie tak i w kodzie programu znajdzie się błąd chciałoby się, aby program wyświetlił naturę tego błędu. Obsługą błędów zajmują się wyjątki (exception), zanim jednak je omówię pokażę jak działają już wbudowane wyjątki:

x = 1 / 0

Powyższy błąd wywoła obsługę wyjątku związanego z dzieleniem przez zero, oto co wyświetli się w konsoli po wykonaniu tego działania:

-------------------------------------------------------------------
ZeroDivisionError                         Traceback (most recent call last)
 in ()
----> 1 x = 1/0

ZeroDivisionError: division by zero

W konsoli ładnie i przejrzyście wyświetlone zostały informacje co poszło nie tak i gdzie poszło nie tak. W tym przypadku wywołany został wyjątek ZeroDivisionError, czyli dzielenie przez zero.

Kolejny przykład działania wbudowanych wyjątków:

y = 10 x = 5 * (1 - y))

Powyższy kod wywoła następujący wyjątek:

In [5]: x = 5 * (1 - y))
  File "", line 1
    x = 5 * (1 - y))
                   ^
SyntaxError: invalid syntax

Tym razem wyjątek to SyntaxError (błąd składni).

Jak widać wyjątki są dość ważne, ponieważ wskazują jaki błąd został popełniony, i gdzie został popełniony co ułatwia eliminację błędu.

Przechwytywanie wyjątków

Wyjątki są przechwytywane za pomocą try, except w następujący przykładowy sposób:

while True: try: k = int(input("Podaj jakąś wartość liczbową: ")) break except ValueError: print("To raczej liczba nie jest. Spróbuj ponownie!")

Przykład działania tego kodu:

Podaj jakąś liczbę: s
To raczej liczba nie jest. Spróbuj ponownie!
Podaj jakąś liczbę: 10

Oczywiście możliwe jest obsłużenie kilku różnych wyjątków, np w następujący sposób:

try: exp = eval(input("Podaj wyrażenie do obliczenia: ")) except ZeroDivisionError as e: print('Dzielisz przez zero ({0})'.format(e)) except NameError as e: print('Nieznana zmienna ({0})'.format(e)) except TypeError as e: print("Błąd typu danych lub deklaracji zmiennej ({0})".format(e), calcText) except: print("Błąd składni") else: print(exp)

W powyższym kodzie wykorzystana została funkcja eval, która jako argument przyjmuje tekst. Tekst ten jest interpretowany jak kod Pythona. Żeby być szczególnie dokładnym chodzi o jednoliniowy kod. Wewnątrz tekstu może znajdować się wszystko co prowadzi do konkluzji, że użytkownik może wpisać jakąś głupotę. Warto więc obsłużyć w odpowiedni sposób wyjątki, co też i powyższy kod czyni.

Oto przykład działania powyższego kodu:

Podaj wyrażenie do obliczenia: 2 * 3 / 0
Dzielisz przez zero (division by zero)

Szczegółowa lista wyjątków, jakie można przechwycić została opisana na stronie docs.python.org. Poniżej zamieszczam listę tychże wyjątków:

Tworzenie własnych wyjątków:

Istnieje też możliwość tworzenia swoich własnych wyjątków, choć zalecane jest wykorzystywanie wyjątków wymienionych na powyższej liście. Oto przykład kodu wykorzystującego własnoręcznie utworzony wyjątek:

class NegativeValue(Exception): pass def silnia(n): if n < 0: raise NegativeValue() s = 1 if n < 2: return 1 for i in range(2, n + 1): s *= i return s try: print(silnia(int(input("Podaj liczbę, której silnię chcesz obliczyć: ")))) except NegativeValue: print("Nie możn aobliczyć silni z ujemnej liczby") except: print("Wyjątek")

W powyższym przykładzie klasa NegativeValue jest klasą obsługi własnego błędu. Taka klasa musi dziedziczyć po klasie Exception. Nasz wyjątek zostanie wywołany wewnątrz funkcji Silnia za pomocą słowa kluczowego raise jako odpowiedź na podanie jako argumentu tej funkcji liczby ujemnej.