Tento návod vysvetľuje spracovanie výnimiek v jazyku Python pomocou bloku Try Except s pomocou príkladov programovania:
Dva typy chýb môžu spôsobiť náhle zastavenie programu Python, t. j. Chyby syntaxe a Výnimky . V tomto učebnom texte sa budeme venovať druhému typu chyby (Výnimky) v rámci niekoľkých dôležitých tém.
Spracovanie výnimiek v našej aplikácii nám prinesie veľa výhod, ako napr:
- Vytvorenie robustnej aplikácie.
- Vytvorenie čistého a bezchybného kódu.
Python Skúste okrem
Jednou z dobrých správ je, že Python má dostatok zabudovaných výnimiek na zachytenie chýb v našom kóde. Taktiež nám dáva možnosť vytvoriť si vlastné výnimky, keď žiadna zo zabudovaných výnimiek nevyhovuje našim potrebám.
Čo je výnimka
Čo je to výnimka v jazyku Python? Zjednodušene povedané, kedykoľvek sa interpret jazyka Python pokúsi vykonať neplatný kód, vyvolá výnimku a v prípadoch, keď takáto výnimka nie je spracovaná, naruší normálny priebeh inštrukcií programu a vypíše spätnú správu.
Vytvorme neplatný kód a pozrime sa, ako bude interpret jazyka Python reagovať.
Otvorte shell jazyka Python a spustite nasledujúci kód.
>>> 50/0
Toto je jedna z najčastejších chýb pri programovaní. Vyššie uvedený kód sa snaží rozdeliť číslo 50 podľa 0 (nula). Prekladač jazyka Python to považuje za neplatnú operáciu a vyvolá chybu ZeroDivisionError , preruší program a vypíše spätnú väzbu.
Jasne vidíme, že ZeroDivisionError je výnimka, ktorá bola vyvolaná. Je to skutočne vlastný spôsob Pythonu, ako nám povedať, že nie je v poriadku deliť číslo nulou. Hoci v iných jazykoch, napríklad v JavaScripte, to nie je chyba; a Python tento postup striktne zakazuje.
Tiež je dôležité vedieť, že toto je len objekt výnimky a Python má mnoho takýchto objektov zabudovaných. Pozrite si túto oficiálnu dokumentáciu Pythonu, kde nájdete všetky zabudované výnimky Pythonu.
Pochopenie spätného sledovania
Skôr ako sa pustíme do spracovania výnimiek, myslím, že nám pomôže pochopiť, čo presne sa stane, ak výnimky nespracujeme, a ako sa Python snaží čo najlepšie nás informovať o našej chybe.
Vždy, keď Python narazí na chybu, vyvolá výnimku. Ak táto výnimka nie je spracovaná, potom vytvorí určitú informáciu nazývanú Traceback.
Obsahuje:
- Chybové hlásenie, ktoré nám hovorí, aká výnimka bola vyvolaná a čo sa stalo pred vyvolaním tejto výnimky.
- Rôzne čísla riadkov kódu, ktoré spôsobili túto chybu. Chyba môže byť spôsobená sekvenciou volaní funkcie tzv. zásobník volaní ktorému sa budeme venovať neskôr.
Hoci je to trochu mätúce, sľubujeme, že ďalší príklad vnesie do nášho chápania viac svetla.
Spomeňte si na spätnú väzbu, ktorá bola vypísaná pri delení 50 číslom 0. Vidíme, že spätná väzba obsahuje nasledujúce informácie:
- Súbor "": To nám hovorí, že tento kód bol spustený z konzolového terminálu.
- riadok 1: To nám hovorí, že chyba sa vyskytla v tomto riadku.
- ZeroDivisionError: rozdelenie podľa nula: Informuje nás o tom, aká výnimka bola vznesená a čo ju spôsobilo.
Skúsme iný príklad a možno uvidíme, ako zásobník volaní Otvorte editor, zadajte nasledujúci kód a uložte ho ako 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
Otvorte terminál v adresári, kde sa nachádza tento súbor, a spustite ho.
python tracebackExp.py
Zobrazí sa nasledujúca spätná väzba:
Vyššie uvedený traceback sa môže zdať mätúci, ale v skutočnosti nie je. Pythonisti prišli na najlepší spôsob čítania tracebacku, ktorý je z zdola nahor . Využime teda túto múdrosť a pokúsme sa pochopiť, čo nám táto spätná väzba ponúka.
- V dolnej časti sa zobrazí výnimka, ktorá bola vyvolaná, a dôvod jej vyvolania.
- Ak sa presunieme nahor, dostaneme názov súboru tracebackExp .py, kde sa táto chyba vyskytla, výpočet, ktorý túto chybu spôsobil compute = numb/div, funkciu stack2 a číslo riadku 6 odkazu, kde sa tento výpočet vykonal.
- Keď sa presunieme nahor, vidíme, že naša funkcia stack2 bola volaná vo funkcii stack1 na riadku číslo 3.
- Ak sa presunieme na najvyšší riadok, vidíme, že funkcia stack1 bola volaná na riadku číslo 11. < modul > nám hovorí, že ide o súbor, ktorý sa vykonáva.
Bežné výnimky jazyka Python
Knižnica Python definuje veľké množstvo vstavaných výnimiek. Môžete si pozrieť dokumentáciu Pythonu alebo zavolať vstavanú funkciu miestne (), ako je uvedené nižšie:
>>> dir(locals()['__builtins__'])
Nebudeme sa snažiť venovať všetkým týmto výnimkám, ale pozrieme sa na niekoľko bežných výnimiek, s ktorými sa pravdepodobne stretnete.
#1) Chyba typu
Vyvolá sa, keď sa operácia alebo funkcia použije na objekt nevhodného typu.
Príklad 1
Uvažujme nasledujúci program. Vezme dividendu a deliteľa, potom vypočíta a vypíše výsledok delenia dividendy deliteľom.
def compute_division(): dividenda = int(input("Zadajte dividendu: ")) # cast string na int deliteľ = input("Zadajte deliteľ: ") # no casting # Vypočítajte delenie výsledok = dividenda/diviteľ # print result print("Výsledok {}/{} je: {}".format(dividenda, deliteľ, výsledok)) if __name__ == '__main__': result = compute_division()
Od používateľa požadujeme hodnotu dividendy a deliteľa, ale zabudli sme premeniť reťazcovú hodnotu deliteľa na celé číslo. Takže nakoniec je typ dividendy integer( int ) a typ deliteľa je string( str ). Potom dostaneme TypeError pretože operátor delenia (/) nepracuje s reťazcami.
Možno vás bude zaujímať, že na rozdiel od jazyka Python má Javascript funkciu Type Coercion, ktorá v podstate konvertuje jeden z typov operandov na ekvivalentnú hodnotu typu druhého operandu, ak sú operandy rôznych typov.
#2) ValueError
Táto funkcia sa zobrazí, keď operácia alebo funkcia dostane argument, ktorý má správny typ, ale nevhodnú hodnotu.
Príklad 2
Zvážte náš program v Príklad 1 vyššie.
Ak používateľ zadá alfanumerickú hodnotu pre dividendu, napríklad "3a", náš program vyvolá výnimku ValueError. Je to preto, že hoci metóda int() v jazyku Python prijíma ľubovoľné číslo alebo reťazec a vracia celočíselný objekt, reťazcová hodnota by nemala obsahovať písmená alebo akúkoľvek nečíselnú hodnotu.
#3) Chyba atribútu
Táto výnimka je vyvolaná pri priraďovaní alebo odkazovaní na atribút, ktorý neexistuje.
Príklad 3
Uvažujme o nasledujúcom programe. Vezme číslo a vypočíta jeho druhú odmocninu pomocou matematického modulu Python
import math # importovať matematickú knižnicu, aby sme získali prístup k jej kódu def compute_square_root(number): # vypočítať druhú odmocninu pomocou matematickej knižnice result = math.sqr(number) return result if __name__ == '__main__': # získať vstup pre výpočet od používateľa number = int(input("Compute Square root of: ")) # zavolať funkciu na výpočet druhej odmocniny
Keď používateľ zadá číslo, náš program sa pokúsi použiť funkciu z modulu math na výpočet jeho druhej odmocniny, ale práve tu sme urobili chybu. Namiesto sqrt sme omylom zadali sqr, ktoré v module math neexistuje.
Snažili sme sa teda odkazovať na atribút sqr, ktorý neexistuje, čo viedlo k vyvolaniu výnimky AttributeError. Väčšina z nás robí tento druh chyby často. Takže nie ste sami.
Spracovanie výnimiek pomocou Try Except
Jednou z vecí, ktorej ako programátori venujeme svoj čas, je písanie robustného kódu, ktorý je odolný. Kód, ktorý sa nezlomí v dôsledku nejakej chyby. V jazyku Python to môžeme dosiahnuť tak, že naše príkazy uzavrieme do vyskúšajte - okrem vyhlásenie.
Príkaz Try-Except v jazyku Python
Príkaz try-except má nasledujúcu štruktúru:
try: #vôj kód patrí sem except """Tu uveďte typ(y) výnimky"""": #spracujte výnimku tu
Uzavrime kód do tracebackExp .py vnútri prí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 tu: # 9 print(tu) # 10 if __name__ == '__main__': # 12 numb = 5 # 13 stack1(numb) # 14 print("program continuous") # 15
Spustením tohto kódu sa vytvorí výstup
Takto funguje príkaz try-except. Python vykoná kód v bloku try riadok 7-8 Ak sa nenájde žiadny neplatný kód, potom sa kód v bloku except riadok 10 sa preskočí a pokračuje sa vo vykonávaní.
Ak sa však nájde neplatný kód, vykonávanie sa okamžite zastaví v bloku try a skontroluje sa, či sa vyvolaná výnimka zhoduje s tou, ktorú sme uviedli v príkaze except riadok 9 . ak sa zhoduje, vykoná sa blok except a pokračuje sa ďalej. ak sa nezhoduje, program sa preruší.
Blok try zvyčajne obsahuje kód, ktorý môže vyvolať výnimku, zatiaľ čo blok except výnimku zachytí a spracuje.
Spracovanie viacerých výnimiek pomocou funkcie Except
Viacero výnimiek môžeme spracovať buď pomocou jedného "except", alebo viacerých "excepts". Všetko závisí od toho, ako chcete jednotlivé výnimky spracovať.
#1) Spracovanie viacerých výnimiek pomocou jedného výnimky
try: #vôj kód patrí sem except(Exception1[, Exception2[,...ExceptionN]]): #vyriešiť výnimku tu
Táto metóda sa používa vtedy, keď predpokladáme, že náš kód môže vyvolať rôzne výnimky, a chceme v každom prípade vykonať rovnakú akciu. Ak teda interpreter jazyka Python nájde zhodu, vykoná sa kód napísaný v bloku except.
Pozrime sa na nasledujúci príklad kódu Pythonu
def get_fraction(value, idx): arr = [4,5,2,0] # zoznam čísel idx_value = arr[idx] # ak idx je> dĺžka arr, vyskočí IndexError value/idx_value # ak idx_value == 0, vyskočí ZeroDivisionError if __name__ =='__main__': # set 'value' and 'idx' value = 54 idx = 3 # call function in a try-except statement. try: result = get_fraction(value, idx) print("Fraction is ", result) except(IndexError, ZeroDivisionError) as ex: print(ex)
Máme dve možné výnimky, ktoré by sa tu mohli uplatniť, ZeroDivisionError a IndexError Ak sa vyskytne niektorá z týchto výnimiek, vykoná sa blok except.
Vo vyššie uvedenom kóde je idx=3, takže idx_ hodnota sa stáva 0 a hodnota /idx_ hodnota vyvolá chybu ZeroDivisionError
#2) Spracovanie viacerých výnimiek s viacerými výnimkami
try: #vaš kód ide sem except Exception1: #handle exception1 here except Exception2: #handle exception2 here except ExceptionN: #handle exceptionN here
Ak by sme chceli spracovať každú výnimku samostatne, môžete to urobiť takto.
Uveďme si príklad kódu Pythonu
def get_fraction(value, idx): arr = [4,5,2,0] # zoznam čísel idx_value = arr[idx] # ak idx je> dĺžka arr, vyskočí IndexError value/idx_value # ak idx_value == 0, vyskočí ZeroDivisionError if __name__ =='__main__': # set 'value' and 'idx' value = 54 idx = 5 # call function in a try-excepts statement. try: result = get_fraction(value, idx) print("Fraction is ", result) exceptIndexError: print("idx {} je mimo rozsahu".format(idx)) except ZeroDivisionError: print("arr[{}] je 0. Preto nemôžeme deliť nulou".format(idx)) except Exception as ex: print(ex) print("Nie je isté, čo sa stalo, takže nie je bezpečné pokračovať, \ aplikácia bude prerušená") raise ex
Všimli sme si, že príkaz Exception bol použitý v poslednom príkaze except. Je to preto, že objekt výnimky Exception zodpovedá akejkoľvek výnimke. Z tohto dôvodu by mal byť vždy posledný, pretože Python prestane kontrolovať ostatné obsluhy výnimiek, keď sa jedna z nich zhoduje.
Vo vyššie uvedenom kóde, idx=5 , teda arr[idx] zvýši IndexError pretože idx je väčšia ako dĺžka zoznamu arr
Taktiež nie je isté, akú výnimku aplikácia vyvolala, nikdy nie je bezpečné pokračovať vo vykonávaní. Preto máme typ Exception, aby sme zachytili všetky nepredvídané výnimky. Potom informujeme používateľa a prerušíme aplikáciu vyvolaním rovnakej výnimky.
Vyskúšajte vyhlásenie Else
Toto je voliteľná funkcia spracovania výnimiek a umožňuje pridať kód, ktorý sa má spustiť, ak sa nevyskytla žiadna chyba. Ak sa vyskytne chyba, tento else-block sa nespustí.
Zoberte si príklad kódu Pythonu nižšie, otvorte si editor a uložte kód ako elseTry.py
def fraction_of_one(divisor): value = 1/divisor # ak je divisor nula, bude vyvolaná chyba ZeroDivisionError return value if __name__ == '__main__': while True: try: # Získajte vstup od používateľa. # ak vstup nie je platným argumentom pre int(), bude vyvolaná chyba ValueError divisor = int(input("Zadajte divisor: ")) # zavolajte našu funkciu na výpočet zlomku value = fraction_of_one(divisor) except (ValueError,ZeroDivisionError): print("Input nemôže byť nula a mal by byť platným literálom pre int(). Prosím, skúste to znova!") else: print("Value: ", value) break
Dostaneme vstup od používateľa a použijeme ho na delenie 1. Máme tu dve možné výnimky, neplatný vstup používateľa, ktorý spôsobí ValueError a nula(0) čo spôsobí. ZeroDivisionError . Náš príkaz except sa zaoberá týmito chybami.
Teraz chceme vypísať hodnotu hodnota . Náš blok else zabezpečuje, že sa vypíše len vtedy, ak sa náš blok try vykoná bez chyby. To je dôležité, pretože ak sa v našom bloku try vyskytne chyba, blok hodnota bude nedefinovaný. Prístup k nemu teda vyvolá ďalšiu chybu.
Spustite vyššie uvedený kód pomocou Python elseTry.py
Vyššie uvedený výstup ukazuje, že pre prvý vstup sme zadali 0 a stlačte ENTER. Keďže náš deliteľ dostal hodnotu 0, 1/deliteľ sa zvýšil zeroDivisionError Naším druhým vstupom bolo k, ktoré je neplatné pre int (), teda výnimka ValueError sa zvýši.
Ale náš posledný vstup bol 9, čo je platné a výsledkom je hodnota " hodnota " vytlačené ako 0.1111111111111111
Skúste konečne vyhlásenie
To je tiež voliteľná funkcia spracovania výnimiek a vždy sa spustí bez ohľadu na to, čo sa stane v obsluhe výnimiek.
To znamená:
- Či sa vyskytne alebo nevyskytne výnimka
- Aj keď sa v iných blokoch volá "return".
- Aj keď je skript ukončený v iných blokoch
Ak teda máme kód, ktorý chceme spustiť vo všetkých situáciách, finally-block je náš človek. Tento blok sa väčšinou používa na upratovanie, napríklad na zatváranie súborov.
Uveďme si príklad kódu Pythonu
def readFile(file_path): try: openFile = open(file_path,'r') # Otvorte súbor len na čítanie print(openFile.readline()) # Prečítajte prvý riadok obsahu súboru except FileNotFoundError as ex: print(ex) finally: print("Čistenie...") openFile.close() if __name__ == '__main__': filePath = './text.txt' readFile(filePath)
Tento kód sa pokúsi otvoriť a prečítať súbor text.txt v aktuálnom adresári. Ak súbor existuje, náš program vypíše prvý riadok súboru a potom sa spustí náš blok finally a súbor sa uzavrie.
Povedzme, že v adresári, kde sa nachádza tento súbor programu, máme súbor s názvom text.txt, ktorý obsahuje Hello. Ak spustíme program, dostaneme výstup
Tento príklad som vybral zámerne, pretože som chcel, aby sme sa venovali malému problému, ktorý môže nastať pri zatváraní súborov v bloku finally.
Ak súbor neexistuje, objaví sa výnimka FileNotFoundError sa zvýši a premenná openFile nebude definovaný a nebude objektom súboru. Preto pokus o jeho uzavretie v bloku finally vyvolá výnimku UnboundLocalError ktorá je podtriedou triedy NameError .
To v podstate znamená, že sa snažíme odkazovať na premennú openFile pred jeho pridelením.
Malý trik spočíva v použití obsluhy výnimiek vo vnútri bloku finally.
def readFile(file_path): try: openFile = open(file_path,'r') # Otvorte súbor len na čítanie print(openFile.readline()) # Prečítajte prvý riadok obsahu súboru except FileNotFoundError as ex: print(ex) finally: try: print("Čistenie...") openFile.close() except: # zachytí všetky výnimky pass # Ignorujte túto chybu, pretože nás nezaujíma. if __name__ == '__main__': filePath = './text.txt' readFile(filePath)
Ak náš try-block vyvolá chybu FileNotFoundError, potom budeme mať nasledujúci výstup
Zvýšenie výnimky
Jednou z dobrých správ o výnimkách v jazyku Python je, že ich môžeme zámerne vyvolávať. Výnimky sa vyvolávajú pomocou príkazu zvýšiť výrok .
Príkaz raise má nasledujúcu syntax:
raise [ExceptionName[(*args: Object)]]
Otvorte terminál a vyvolajte ľubovoľný objekt výnimky zo zabudovaných výnimiek jazyka Python. Napríklad, ak sa vyskytne chyba ZeroDivisionError:
>>> raise ZeroDivisionError("Nemožno deliť nulou")
Získame spätnú väzbu:
Prečo je teda dôležité zvýšiť počet výnimiek?
- Pri práci s vlastnými výnimkami.
- Počas kontroly príčetnosti.
Vlastné triedy výnimiek
Vlastná výnimka je výnimka, ktorú vytvoríte na spracovanie chýb, ktoré sú špecifické pre vašu potrebu. Trik spočíva v tom, že definujeme triedu, ktorá je odvodená od objektu Výnimka , potom použijeme príkaz raise na vyvolanie našej triedy výnimiek.
Predpokladajme, že chceme skontrolovať vstup používateľa a uistiť sa, že vstupná hodnota nie je záporná (kontrola správnosti). Samozrejme, mohli by sme vyvolať výnimku Pythonu ValueError, ale radi by sme si chybu prispôsobili tým, že jej dáme špecifický a zrozumiteľný názov, ako napr. InputIsNegativeError . Táto výnimka však nie je výnimka zabudovaná v jazyku Python.
Najprv teda vytvoríme našu základnú triedu, ktorá bude odvodená od triedy Exception.
class CustomError(Exception): "Výnimka základnej triedy pre všetky výnimky tohto modulu" pass
Potom vytvoríme našu triedu výnimiek, ktorá bude dediť základnú triedu a bude spracovávať našu špecifickú chybu.
class InputIsNegativeError(CustomError): """Vyvolaná, keď používateľ zadá zápornú hodnotu""" pass
Otestujme 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")
Vyššie uvedený kód si vyžiada vstup od používateľa a skontroluje, či je záporný. Ak je pravdivý, vyvolá našu vlastnú výnimku InputIsNegativeError, ktorá sa neskôr zachytí v príkaze except.
Nižšie je uvedený kompletný kód:
class CustomError(Exception): "Výnimka základnej triedy pre všetky výnimky tohto modulu" pass class InputIsNegativeError(CustomError): """Vyvolaná, keď používateľ zadá zápornú hodnotu"" pass if __name__ == '__main__': try: value = int(input("Zadajte číslo: ")) if value <0: raise InputIsNegativeError # Raise exception if value is negative except InputIsNegativeError: # catch and handle exceptionprint("Vstupná hodnota by nemala byť záporná")
Ak je vstupná hodnota záporné číslo, napríklad -1, potom budeme mať výstup:
Viac informácií o vlastných výnimkách Pythonu nájdete v dokumentácii Python.
Často kladené otázky
Otázka č. 1) Ako Python rieši výnimku?
Odpoveď: Python spracováva výnimky pomocou príkazu príkaz try-except Kód, ktorý môže vyvolať výnimku, je umiestnený a vykonaný v skúste blok zatiaľ čo okrem bloku obsahuje kód, ktorý bude spracovávať výnimky, ak sa nejaké vyskytnú.
Q #2) Čo je to vyvolanie výnimky v jazyku Python?
Odpoveď: Vždy, keď interpreter jazyka Python narazí na neplatný kód, vyvolá výnimku, čo je vlastný spôsob jazyka Python, ako nám oznámiť, že sa stalo niečo neočakávané. Výnimky môžeme vyvolávať aj zámerne pomocou príkazu zvýšiť výrok .
Q #3) Ako Python rieši viacnásobné výnimky?
Odpoveď: Python spracúva viacero výnimiek pomocou jedného bloku except alebo viacerých blokov except.
V prípade jedného bloku sa výnimky odovzdávajú ako tuple: okrem (Exception1, Exception2,..,ExceptionN) a Python kontroluje zhodu sprava doľava. V tomto prípade sa pre každú výnimku vykoná rovnaká akcia.
Ďalším spôsobom, ako zachytiť všetky výnimky, je vynechať názov výnimky za kľúčovým slovom except.
except: # tu sa spracujú všetky výnimky
Druhým spôsobom je použitie bloku except pre každú výnimku:
except Exception1: # kód na spracovanie Exception1 ide sem except Exception2: # kód na spracovanie Exception2 ide sem except ExceptionN: # kód na spracovanie ExceptionN ide sem
Takto môžete vykonať samostatné akcie pre každú výnimku.
Q #4) Prečo je v jazyku Python dôležité spracovanie výnimiek?
Odpoveď: Výhodou spracovania výnimiek v jazyku Python je, že môžeme vytvárať robustné, čisté a bezchybné aplikácie. Nebudeme chcieť, aby náš produkčný kód spadol kvôli nejakej chybe, preto chyby spracujeme a udržíme našu aplikáciu v chode.
Otázka č. 5) Ako v jazyku Python ignorujete výnimku?
Odpoveď: Ak chcete ignorovať výnimku v jazyku Python, použite príkaz odovzdať kľúčové slovo v bloku except. Povedzme, že chceme ignorovať výnimku ValueError. Urobíme to týmto spôsobom:
okrem ValueError: pass
Pokiaľ neviete, čo robíte, je zlým zvykom ignorovať výnimky. O všetkých potenciálnych chybách používateľa aspoň informujte.
Záver
V tomto učebnom texte sme sa venovali: výnimkám Pythonu, spätnému sledovaniu; ako spracovať výnimky pomocou Vyskúšajte stránku / Okrem / Inak / Konečne bloky, ako Zvýšiť Výnimky a nakoniec ako vytvoriť vlastné výnimky.
Ďakujeme za prečítanie!