Python Try Except - Python Handling Exception With Examples

Tento výukový kurz vysvětluje zpracování výjimek v jazyce Python pomocí bloku Try Except s pomocí příkladů programování:

Dva typy chyb mohou způsobit náhlé zastavení programu Python, tj. Chyby syntaxe a Výjimky . V tomto tutoriálu se budeme zabývat druhým typem chyby (Výjimky) v rámci několika důležitých témat.

Zpracování výjimek v naší aplikaci nám přinese velké výhody, např.:

  • Vytvoření robustní aplikace.
  • Vytvoření čistého a bezchybného kódu.

Python Try Except

Jednou z dobrých zpráv je, že Python má velké množství vestavěných výjimek pro zachycení chyb v našem kódu. Také nám dává možnost vytvářet vlastní výjimky, pokud žádná z vestavěných výjimek nevyhovuje našim potřebám.

Co je výjimka

Co je to výjimka v jazyce Python? Zjednodušeně řečeno, kdykoli se interpret jazyka Python pokusí provést neplatný kód, vyvolá výjimku a v případech, kdy takovou výjimku nezvládne, naruší normální průběh instrukcí programu a vypíše zpětnou stopu.

Vytvořme neplatný kód a podívejme se, jak bude interpret jazyka Python reagovat.

Otevřete shell Pythonu a spusťte následující kód.

 >>> 50/0 

Jedná se o jednu z nejčastějších chyb při programování. Výše uvedený kód se snaží vydělit číslo 50 podle 0 (nula). Překladač Pythonu to považuje za neplatnou operaci a vyvolá chybu. ZeroDivisionError , přeruší program a vypíše zpětnou stopu.

Jasně vidíme, že ZeroDivisionError je výjimka, která byla vyvolána. Je to skutečně způsob, jak nám Python sděluje, že není v pořádku dělit číslo nulou. Ačkoli v jiných jazycích, jako je JavaScript, to není chyba; a Python tuto praxi striktně zakazuje.

Je také důležité vědět, že se jedná pouze o objekt výjimky a že Python má mnoho takových objektů vestavěných. Podívejte se na tuto oficiální dokumentaci Pythonu, kde najdete všechny vestavěné výjimky Pythonu.

Porozumění zpětnému sledování

Než se pustíme do ošetřování výjimek, myslím, že nám pomůže pochopit, co přesně se stane, když výjimky neošetříme, a jak se Python snaží informovat nás o naší chybě.

Kdykoli Python narazí na chybu, vyvolá výjimku. Pokud tuto výjimku nezvládne, vytvoří informaci nazvanou Traceback (zpětné sledování).

Obsahuje:

  • Chybová zpráva, která nám říká, jaká výjimka byla vyvolána a co se stalo před vyvoláním této výjimky.
  • Různá čísla řádků kódu, který tuto chybu způsobil. Chyba může být způsobena sekvencí volání funkce tzv. zásobník volání o kterém se zde zmíníme později.

I když je to trochu matoucí, slibujeme, že další příklad vnese do našeho chápání více světla.

Vzpomeňte si na zpětnou vazbu, která byla vypsána při dělení 50 nulou výše, a uvidíme, že zpětná vazba obsahuje následující informace:

  • Soubor "": To nám říká, že tento kód byl spuštěn z konzolového terminálu.
  • řádek 1: To nám říká, že chyba nastala v tomto řádku.
  • ZeroDivisionError: rozdělení podle nula: Řekne nám, jaká výjimka byla vznesena a co ji způsobilo.

Vyzkoušejme další příklad a podívejme se, jak se zásobník volání Otevřete editor, zadejte níže uvedený kód a uložte jej jako tracebackExp .py

 def stack1(numb): # 1 div = 0 # 2 stack2(numb, div) # 3 def stack2(numb, div): # 5 compute = numb/div # 6 print(compute) # 7 if __name__ == '__main__': # 9 numb = 5 # 10 stack1(numb) # 11 

Otevřete terminál v adresáři, kde se tento soubor nachází, a spusťte jej.

 python tracebackExp.py 

Zobrazí se následující zpětná vazba:

Výše uvedený traceback se může zdát matoucí, ale ve skutečnosti tomu tak není. Pythonisté přišli s nejlepším způsobem, jak číst traceback, který je z příkazu zdola nahoru . Využijme tedy této moudrosti a pokusme se pochopit, co nám tato zpětná vazba nabízí.

  • V dolní části se zobrazí výjimka, která byla vyvolána, a důvod jejího vyvolání.
  • Pokud se přesuneme nahoru, získáme název souboru tracebackExp .py, kde k této chybě došlo, výpočet, který tuto chybu způsobil compute = numb/div, funkci stack2 a číslo řádku 6 odkazu, kde byl tento výpočet proveden.
  • Když se přesuneme nahoru, vidíme, že naše funkce stack2 byla volána ve funkci stack1 na řádku číslo 3.
  • Přesuneme-li se na nejvyšší řádek, vidíme, že funkce stack1 byla volána na řádku číslo 11. < modul > nám říká, že se jedná o soubor, který se spouští.

Běžné výjimky jazyka Python

Knihovna Pythonu definuje velké množství vestavěných výjimek. Můžete se podívat do dokumentace Pythonu nebo zavolat vestavěnou funkci místní (), jak je uvedeno níže:

 >>> dir(locals()['__builtins__']) 

Nebudeme se snažit zabývat všemi těmito výjimkami, ale podíváme se na několik běžných výjimek, se kterými se pravděpodobně setkáte.

#1) Chyba typu

Je vyvolána, když je operace nebo funkce použita na objekt nevhodného typu.

Příklad 1

Uvažujme následující program. Přijme dividendu a dělitele, poté vypočítá a vypíše výsledek dělení dividendy dělitelem.

 def compute_division(): dividenda = int(input("Zadejte dividendu: ")) # cast řetězce na int dělitel = input("Zadejte dělitele: ") # bez castu # Vypočtěte dělení výsledek = dividenda/dělitel # vytiskněte výsledek print("Výsledek {}/{} je: {}".format(dividenda, dělitel, výsledek)) if __name__ == '__main__': result = compute_division() 

Od uživatele si vyžádáme hodnotu dividendy a dělitele, ale zapomeneme převést řetězcovou hodnotu dělitele na celé číslo. Takže nakonec bude typ dividendy integer( int ) a typ dělitele je string( str ). Pak získáme TypeError protože operátor dělení (/) nepracuje s řetězci.

Možná vás bude zajímat, že na rozdíl od jazyka Python má Javascript funkci Type Coercion, která v podstatě převádí jeden z typů operandů na ekvivalentní hodnotu typu druhého operandu, pokud jsou operandy různých typů.

#2) ValueError

Tato situace nastane, když operace nebo funkce obdrží argument správného typu, ale s nevhodnou hodnotou.

Příklad 2

Zvažte náš program v Příklad 1 výše.

Pokud uživatel zadá pro dividendu alfanumerickou hodnotu, například '3a', pak náš program vyvolá výjimku ValueError. Je to proto, že ačkoli metoda Pythonu int() přijímá libovolné číslo nebo řetězec a vrací celočíselný objekt, řetězcová hodnota by neměla obsahovat písmena nebo jakoukoli jinou než číselnou hodnotu.

#3) Chyba atributu

Tato výjimka je vyvolána při přiřazování nebo odkazování na neexistující atribut.

Příklad 3

Vezměme si následující program, který přijme číslo a vypočítá jeho druhou odmocninu pomocí matematického modulu Pythonu.

 import math # importujte matematickou knihovnu, abyste získali přístup k jejímu kódu def compute_square_root(number): # vypočtěte odmocninu pomocí matematické knihovny result = math.sqr(number) return result if __name__ == '__main__': # získejte vstup pro výpočet od uživatele number = int(input("Compute Square root of: ")) # zavolejte funkci pro výpočet odmocniny. 

Když uživatel zadá číslo, náš program se pokusí použít funkci z matematického modulu pro výpočet jeho druhé odmocniny, ale právě zde jsme udělali chybu. Místo sqrt jsme omylem zadali sqr, které v matematickém modulu neexistuje.

Snažili jsme se tedy odkázat na atribut sqr, který neexistuje, což vedlo k vyvolání výjimky AttributeError. Většina z nás dělá podobné chyby často. Nejste tedy sami.

Zpracování výjimek pomocí Try Except

Jako programátor většina z nás věnuje svůj čas psaní robustního kódu, který je odolný. Kód, který se nerozbije kvůli nějaké chybě. V jazyce Python toho můžeme dosáhnout tím, že své příkazy uzavřeme uvnitř příkazů typu zkuste - kromě prohlášení.

Příkaz Try-Except v jazyce Python

Příkaz try-except má následující strukturu:

 try: #vůj kód patří sem except """Zde uveďte typ(y) výjimky"""": #zpracujte výjimku zde 

Uzavřeme kód do tracebackExp .py uvnitř příkazu try-except.

 def stack1(numb): # 1 div = 0 # 2 stack2(numb, div) # 3 def stack2(numb, div): # 5 try: # 6 compute = numb/div # 7 print(compute) # 8 except ZeroDivisionError as zde: # 9 print(zde) # 10 if __name__ == '__main__': # 12 numb = 5 # 13 stack1(numb) # 14 print("program continuous") # 15 

Spuštěním tohoto kódu získáte výstup

Takto funguje příkaz try-except. Python provede kód v bloku try. řádek 7-8 . Pokud není nalezen žádný neplatný kód, pak se kód v bloku except řádek 10 se přeskočí a pokračuje se v provádění.

Pokud je však nalezen neplatný kód, pak se provádění okamžitě zastaví v bloku try a zkontroluje se, zda se vyvolaná výjimka shoduje s tou, kterou jsme uvedli v příkazu except. řádek 9 . Pokud se shoduje, provede se blok except a pokračuje se dál. Pokud se neshoduje, program se přeruší.

Blok try obvykle obsahuje kód, který může vyvolat výjimku, zatímco blok except výjimku zachytí a zpracuje.

Zpracování více výjimek pomocí příkazu Except

Více výjimek můžeme zpracovat buď pomocí jednoho "except", nebo více "excepts". Záleží na tom, jak chcete jednotlivé výjimky zpracovat.

#1) Zpracování více výjimek pomocí jednoho výjimky

 try: #všechen kód patří sem except(Exception1[, Exception2[,...ExceptionN]]): #zpracování výjimky zde 

Tuto metodu používáme v případě, že předpokládáme, že náš kód může vyvolat různé výjimky, a chceme v každém případě provést stejnou akci. Pokud tedy interpret jazyka Python najde shodu, provede se kód zapsaný v bloku except.

Podívejme se na níže uvedený příklad kódu Pythonu

 def get_fraction(value, idx): arr = [4,5,2,0] # seznam čísel idx_value = arr[idx] # pokud idx je> délka arr, bude vyvolána chyba IndexError value/idx_value # pokud idx_value == 0, bude vyvolána chyba ZeroDivisionError if __name__ =='__main__': # nastav 'value' a 'idx' value = 54 idx = 3 # volání funkce v příkazu try-except. try: result = get_fraction(value, idx) print("Fraction is ", result) except(IndexError, ZeroDivisionError) as ex: print(ex) 

Máme dvě možné výjimky, které by zde mohly být vzneseny, ZeroDivisionError a IndexError . Pokud je některá z těchto výjimek vyvolána, provede se blok except.

Ve výše uvedeném kódu je idx=3, takže idx_ hodnota nabývá hodnoty 0 a hodnota /idx_ hodnota vyvolá chybu ZeroDivisionError

#2) Obsluha více výjimek pomocí více výjimek

 try: #vůj kód patří sem except Exception1: #handle exception1 here except Exception2: #handle exception2 here except ExceptionN: #handle exceptionN here 

Pokud bychom raději chtěli zpracovávat každou výjimku zvlášť, můžete to udělat takto.

Vezměme si níže uvedený příklad kódu Pythonu

 def get_fraction(value, idx): arr = [4,5,2,0] # seznam čísel idx_value = arr[idx] # pokud idx je> délka arr, bude vyvolána chyba IndexError value/idx_value # pokud idx_value == 0, bude vyvolána chyba ZeroDivisionError if __name__ =='__main__': # nastav 'value' a 'idx' value = 54 idx = 5 # volání funkce v příkazu try-excepts. try: result = get_fraction(value, idx) print("Fraction is ", result) exceptIndexError: print("idx {} je mimo rozsah".format(idx)) except ZeroDivisionError: print("arr[{}] je 0. Proto nelze dělit nulou".format(idx)) except Exception as ex: print(ex) print("Nejsem si jistý, co se stalo, takže není bezpečné pokračovat, \ aplikace bude přerušena") raise ex 

Všimli jsme si, že v posledním příkazu except byla použita výjimka Exception. Je to proto, že objekt výjimky Exception odpovídá jakékoliv výjimce. Z tohoto důvodu by měl být vždy poslední, protože Python přestane kontrolovat ostatní obsluhy výjimek, jakmile jedna odpovídá.

Ve výše uvedeném kódu, idx=5 , a proto arr[idx] zvýší IndexError protože idx je větší než délka seznamu arr

Také není jisté, jakou výjimku aplikace vyvolala, nikdy není bezpečné pokračovat ve vykonávání. Proto máme typ Exception, abychom zachytili všechny nepředvídané výjimky. Pak informujeme uživatele a přerušíme aplikaci vyvoláním stejné výjimky.

Příkaz Try Else

Jedná se o volitelná funkce zpracování výjimek a umožňuje přidat kód, který se má spustit, pokud nedošlo k žádné chybě. Pokud dojde k chybě, tento else-block se nespustí.

Vezměte si níže uvedený příklad kódu Pythonu, otevřete editor a uložte kód jako elseTry.py.

 def fraction_of_one(divisor): value = 1/divisor # pokud je divisor nulový, bude vyvolána chyba ZeroDivisionError return value if __name__ == '__main__': while True: try: # Získejte vstup od uživatele. # pokud vstup není platným argumentem pro int(), bude vyvolána chyba ValueError divisor = int(input("Zadejte dělitel: ")) # zavolejte naši funkci pro výpočet zlomku value = fraction_of_one(divisor) except (ValueError,ZeroDivisionError): print("Input nemůže být nula a měl by být platným literálem pro int(). Prosím, zkuste to znovu!") else: print("Value: ", value) break 

Získáme vstup od uživatele a použijeme ho k dělení 1. Máme zde dvě možné výjimky, neplatný vstup od uživatele, který způsobí ValueError a nula(0) což způsobí, že ZeroDivisionError . Tyto chyby řeší náš příkaz except.

Nyní chceme vypsat hodnotu hodnota . Náš else-block zajišťuje, že bude vypsán pouze v případě, že náš try-block proběhne bez chyby. To je důležité, protože pokud dojde k chybě v našem try-bloku, bude se vypisovat hodnota bude nedefinovaný. Přístup k němu tedy vyvolá další chybu.

Spusťte výše uvedený kód pomocí Pythonu elseTry.py

Výstup výše ukazuje, že pro první vstup jsme zadali. 0 a stiskl ENTER. Protože náš dělitel získal hodnotu 0, 1/dělitel se zvýšil o zeroDivisionError Druhým vstupem bylo k, které je neplatné pro int (), tedy výjimka ValueError se zvýší.

Ale náš poslední vstup byl 9, což je platné, a výsledkem je hodnota " hodnota " vytištěno jako 0.111111111111111111

Zkuste konečně prohlášení

To je také volitelná funkce zpracování výjimek a bude vždy spuštěn bez ohledu na to, co se stane v obsluze výjimek.

To znamená:

  • Zda dojde k výjimce, nebo ne
  • A to i v případě, že je v ostatních blocích volán "return".
  • I když je skript ukončen v jiných blocích

Pokud tedy máme kód, který chceme spouštět ve všech situacích, je finally-block náš člověk. Tento blok se většinou používá pro úklid, například pro zavírání souborů.

Vezměme si níže uvedený příklad kódu Pythonu

 def readFile(file_path): try: openFile = open(file_path,'r') # Otevřete soubor pouze pro čtení print(openFile.readline()) # Přečtěte první řádek obsahu souboru except FileNotFoundError as ex: print(ex) finally: print("Čištění...") openFile.close() if __name__ == '__main__': filePath = './text.txt' readFile(filePath) 

Tento kód se pokusí otevřít a přečíst soubor text.txt v aktuálním adresáři. Pokud soubor existuje, pak náš program vypíše první řádek souboru a poté se spustí náš blok finally a soubor uzavře.

Řekněme, že v adresáři, kde se nachází tento programový soubor, máme soubor s názvem text.txt, který obsahuje Hello. Pokud program spustíme, dostaneme výstupní text

Tento příklad jsem zvolil záměrně, protože jsem chtěl, abychom se zabývali malým problémem, který může nastat při uzavírání souborů v bloku finally.

Pokud soubor neexistuje, je výjimka FileNotFoundError a proměnná openFile nebude definován a nebude objektem souboru. Proto pokus o jeho uzavření v bloku finally vyvolá výjimku. UnboundLocalError která je podtřídou třídy NameError .

To v podstatě říká, že se snažíme odkazovat na proměnnou openFile před jeho přidělením.

Malým trikem je použití obsluhy výjimek uvnitř bloku finally.

 def readFile(file_path): try: openFile = open(file_path,'r') # Otevřete soubor pouze pro čtení print(openFile.readline()) # Přečtěte první řádek obsahu souboru except FileNotFoundError as ex: print(ex) finally: try: print("Čištění...") openFile.close() except: # zachytí všechny výjimky pass # Ignorujte tuto chybu, protože nás nezajímá. if __name__ == '__main__': filePath = './text.txt' readFile(filePath) 

Pokud náš try-block vyvolá chybu FileNotFoundError, dostaneme následující výstup.

Zvyšování výjimky

Jednou z dobrých zpráv o výjimkách Pythonu je, že je můžeme záměrně vyvolat. Výjimky se vyvolávají příkazem zvýšit výrok .

Příkaz raise má následující syntaxi:

 raise [ExceptionName[(*args: Object)]] 

Otevřete terminál a vyvolejte libovolný objekt výjimky z integrovaných výjimek jazyka Python. Například, pokud vyvoláme chybu ZeroDivisionError:

 >>> raise ZeroDivisionError("Nelze dělit nulou") 

Získáme zpětnou vazbu:

Proč je tedy důležité zvyšovat výjimky?

  • Při práci s vlastními výjimkami.
  • Během kontroly příčetnosti.

Vlastní třídy výjimek

Vlastní výjimka je výjimka, kterou vytvoříte pro zpracování chyb, které jsou specifické pro vaši potřebu. Trik spočívá v tom, že definujeme třídu, která je odvozena od objektu Výjimka , pak pomocí příkazu raise vyvoláme naši třídu výjimek.

Předpokládejme, že chceme zkontrolovat uživatelský vstup a ujistit se, že vstupní hodnota není záporná (kontrola správnosti). Samozřejmě bychom mohli vyvolat výjimku Pythonu ValueError, ale rádi bychom tuto chybu přizpůsobili tím, že jí dáme konkrétní a srozumitelný název, např. InputIsNegativeError . Tato výjimka však není vestavěnou výjimkou jazyka Python.

Nejprve tedy vytvoříme naši základní třídu, která bude odvozena od třídy Exception.

 class CustomError(Exception): "Základní třída výjimek pro všechny výjimky tohoto modulu" pass 

Poté vytvoříme naši třídu výjimek, která bude dědit základní třídu a bude zpracovávat naši konkrétní chybu.

 class InputIsNegativeError(CustomError): """Vyvoláno, když uživatel zadá zápornou hodnotu""" pass 

Vyzkoušejme to

 try: value = int(input()) if value <0: raise InputIsNegativeError # Raise exception if value is negative except InputIsNegativeError: # catch and handle exception print("Input value shouldn't be negative") 

Výše uvedený kód si vyžádá uživatelský vstup a zkontroluje, zda je záporný. Pokud je pravdivý, vyvolá naši vlastní výjimku InputIsNegativeError, která se později zachytí ve větě except.

Níže je uveden kompletní kód:

 class CustomError(Exception): "Základní třída výjimek pro všechny výjimky tohoto modulu" pass class InputIsNegativeError(CustomError): """Raised when User enters a negative value""" pass if __name__ == '__main__': try: value = int(input("Input a number: ")) if value <0: raise InputIsNegativeError # Raise exception if value is negative except InputIsNegativeError: # catch and handle exceptionprint("Vstupní hodnota by neměla být záporná") 

Pokud je vstupní hodnotou záporné číslo, například -1, pak bude výstupem:

Další podrobnosti o vlastních výjimkách Pythonu najdete v dokumentaci Pythonu.

Často kladené otázky

Q #1) Jak Python řeší výjimku?

Odpověď: Python zpracovává výjimky pomocí příkazu příkaz try-except . Kód, který může vyvolat výjimku, je umístěn a proveden v příkazu zkuste blok zatímco kromě bloku obsahuje kód, který bude zpracovávat případné výjimky.

Q #2) Co je to vyvolání výjimky v jazyce Python?

Odpověď: Kdykoli interpret jazyka Python narazí na neplatný kód, vyvolá výjimku, což je způsob, jakým nám Python sděluje, že se stalo něco neočekávaného. Výjimky můžeme vyvolat i záměrně pomocí příkazu zvýšit výrok .

Q #3) Jak Python řeší vícenásobné výjimky?

Odpověď: Python zpracovává vícenásobné výjimky buď pomocí jednoho bloku except, nebo více bloků except.

V případě jednoho bloku se výjimky předávají jako tuple: kromě (Exception1, Exception2,..,ExceptionN) a Python kontroluje shodu zprava doleva. V tomto případě je pro každou výjimku provedena stejná akce.

Dalším způsobem, jak zachytit všechny výjimky, je vynechat název výjimky za klíčovým slovem except.

 except: # zde zpracováváme všechny výjimky 

Druhým způsobem je použití bloku except pro každou výjimku:

 except Exception1: # kód pro zpracování Exception1 jde sem except Exception2: # kód pro zpracování Exception2 jde sem except ExceptionN: # kód pro zpracování ExceptionN jde sem 

Tímto způsobem můžete pro každou výjimku provést samostatné akce.

Q #4) Proč je v jazyce Python důležité zpracování výjimek?

Odpověď: Výhodou ošetřování výjimek v jazyce Python je, že můžeme vytvářet robustní, čisté a bezchybné aplikace. Nebudeme chtít, aby náš produkční kód kvůli nějaké chybě spadl, takže chyby ošetříme a udržíme naši aplikaci v chodu.

Q #5) Jak v Pythonu ignorujete výjimku?

Odpověď: Chcete-li v jazyce Python ignorovat výjimku, použijte příkaz předat Řekněme, že chceme ignorovat výjimku ValueError. Uděláme to takto:

 except ValueError: pass 

Pokud nevíte, co děláte, je špatným zvykem výjimky ignorovat. O všech možných chybách uživatele alespoň informujte.

Závěr

V tomto kurzu jsme se zabývali: výjimkami Pythonu, zpětným sledováním; jak zpracovávat výjimky pomocí nástroje Zkuste / Kromě / Jinak / Konečně bloky, jak Zvyšte Výjimky a nakonec jak vytvořit vlastní výjimky.

Děkujeme za přečtení!

Posunout nahoru