Deze tutorial geeft uitleg over Exception Handling in Python met behulp van het Try Except-blok met behulp van programmeervoorbeelden:
Twee fouttypes kunnen ervoor zorgen dat een Python-programma abrupt stopt, nl. Syntaxfouten en Uitzonderingen In deze tutorial bespreken we het tweede fouttype (Exceptions) onder verschillende belangrijke onderwerpen.
We zullen veel baat hebben bij het behandelen van uitzonderingen in onze toepassing zoals:
- Een robuuste toepassing maken.
- Een schone en foutloze code maken.
Python Probeer Behalve
Het goede nieuws is dat Python een flink aantal ingebouwde uitzonderingen heeft om fouten in onze code op te vangen. Ook geeft het ons de mogelijkheid om aangepaste uitzonderingen te maken wanneer geen van de ingebouwde uitzonderingen aan onze behoeften voldoet.
Wat is een uitzondering?
Wat is een uitzondering in Python? Wel, eenvoudig gezegd, telkens wanneer de Python-interpreter ongeldige code probeert uit te voeren, roept hij een uitzondering op, en in de gevallen waarin zo'n uitzondering niet wordt behandeld, verstoort hij de normale stroom van de instructies van het programma en drukt hij een traceback af.
Laten we een ongeldige code maken en kijken hoe de Python-interpreter zal reageren.
Open een Python-shell en voer de volgende code uit.
>>> 50/0
Dit is een van de meest voorkomende fouten bij het programmeren. De bovenstaande code probeert het getal 50 door 0 (De Python-interpreter ziet dit als een ongeldige bewerking en geeft een ZeroDivisionError verstoort het programma, en drukt een traceback af.
We kunnen duidelijk zien dat ZeroDivisionError Het is inderdaad Python's eigen manier om ons te vertellen dat het niet cool is om een getal door nul te delen. Hoewel dit in andere talen zoals JavaScript geen fout is; en python deze praktijk strikt verbiedt.
Het is ook belangrijk te weten dat dit slechts een uitzonderingsobject is en dat Python veel van dergelijke objecten heeft ingebouwd. Bekijk de officiële documentatie van Python om alle ingebouwde uitzonderingen van Python te zien.
Traceback begrijpen
Voordat we ingaan op het behandelen van uitzonderingen, denk ik dat het helpt om te begrijpen wat er precies gebeurt als uitzonderingen niet worden behandeld en hoe Python zijn best doet om ons te informeren over onze fout.
Wanneer Python een fout tegenkomt, roept het een uitzondering op. Als deze uitzondering niet wordt afgehandeld, produceert het informatie die Traceback wordt genoemd. Welke informatie bevat deze Traceback?
Het bevat:
- De foutmelding die ons vertelt welke uitzondering werd gemaakt en wat er gebeurde voordat deze uitzondering werd gemaakt.
- De verschillende regelnummers van de code die deze fout heeft veroorzaakt. Een fout kan worden veroorzaakt door een opeenvolging van functieaanroepen die een oproepstapel die we hier later zullen bespreken.
Hoewel het een beetje verwarrend is, beloven we dat het volgende voorbeeld meer licht zal brengen in ons begrip.
Denk aan de traceback die werd afgedrukt bij het delen van 50 door 0 hierboven, we kunnen zien dat de traceback de volgende informatie bevat:
- Bestand "": Dit vertelt ons dat deze code werd uitgevoerd vanaf een console terminal.
- regel 1: Dit vertelt ons dat de fout is opgetreden in dit regelnummer.
- ZeroDivisionError: divisie door nul: Het vertelt ons welke uitzondering werd gemaakt en wat de oorzaak was.
Laten we een ander voorbeeld proberen en misschien zien hoe een oproepstapel Open een editor, voer onderstaande code in en sla op als 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
Open een terminal in de directory waar dit bestand staat en voer het uit.
python tracebackExp.py
U ziet de volgende traceback:
De bovenstaande traceback kan verwarrend lijken, maar dat is het eigenlijk niet. Pythonistas bedachten de beste manier om traceback te lezen, namelijk vanuit de bottom-up Dus, laten we deze wijsheid gebruiken om te proberen te begrijpen wat deze traceback te bieden heeft.
- Onderaan staat de uitzondering die werd gemaakt en waarom die werd gemaakt.
- Als we naar boven gaan, krijgen we de bestandsnaam tracebackExp .py waar deze fout optrad, de berekening die deze fout veroorzaakte compute = numb/div, de functie stack2, en het linknummer regel 6 waar deze berekening werd uitgevoerd.
- We zien dat onze functie stack2 is aangeroepen in de functie stack1 op regel 3.
- Als we naar de bovenste gaan, zien we dat de functie stack1 werd aangeroepen in regel nummer 11. < module > vertelt ons dat het het bestand is dat wordt uitgevoerd.
Veel voorkomende Python-uitzonderingen
De Python bibliotheek definieert een heleboel ingebouwde uitzonderingen. U kunt de Python documentatie raadplegen of de ingebouwde lokale () functie zoals hieronder:
>>> dir(locals()['__builtins__'])
We zullen niet proberen al deze uitzonderingen te behandelen, maar we zullen een paar veel voorkomende uitzonderingen bekijken die u waarschijnlijk zult tegenkomen.
#1) TypeError
Deze wordt opgewekt wanneer een bewerking of functie wordt toegepast op een object van een onjuist type.
Voorbeeld 1
Beschouw het volgende programma: het neemt een dividend en een deler, berekent en drukt het resultaat af van het delen van het dividend door de deler.
def compute_division(): dividend = int(input("Voer het dividend in: ")) # cast string naar int divisor = input("Voer de divisor in: ") # no casting # Compute division result = dividend/divisor # print result print("Het resultaat van {}/{} is: {}".format(dividend, divisor, result)) if __name__ == '__main__': result = compute_division()
We vragen de waarde van het dividend en de deler op bij de gebruiker, maar we vergeten de stringwaarde van de deler om te zetten in een geheel getal. Dus eindigt het dividend met het type geheel getal( int ) en het type deler is string( str ). We krijgen dan de TypeError omdat de deler (/) niet op strings werkt.
Het is misschien interessant om te weten dat Javascript, in tegenstelling tot Python, Type Coercion heeft, dat in principe een van de operand-types converteert naar een equivalente waarde van het andere operand-type wanneer de operand-types verschillend zijn.
#2) ValueError
Dit wordt veroorzaakt wanneer een bewerking of functie een argument ontvangt dat het juiste type heeft, maar een onjuiste waarde.
Voorbeeld 2
Denk aan ons programma in Voorbeeld 1 boven.
Indien de gebruiker een alfanumerieke waarde invoert voor het dividend zoals '3a', dan zal ons programma een ValueError-uitzondering opleveren. Dit komt omdat, hoewel de Python int() methode een willekeurig getal of een willekeurige tekenreeks aanneemt en een geheel getal teruggeeft, de tekenreeks geen letters of andere niet-numerieke waarden mag bevatten.
#3) AttributeError
Deze uitzondering wordt gemaakt bij het toewijzen van of verwijzen naar een attribuut dat niet bestaat.
Voorbeeld 3
Beschouw het onderstaande programma. Het neemt een getal en berekent de vierkantswortel ervan met behulp van de Python wiskunde module
import math # import math library to gain access to its code def compute_square_root(number): # bereken de vierkantswortel met behulp van de math library result = math.sqr(number) return result if __name__ == '__main__': # get input to compute from user number = int(input("Compute Square root of: ")) # call function to compute square root
Wanneer een gebruiker een getal invoert, probeert ons programma een functie uit de wiskunde module te gebruiken om de vierkantswortel ervan te berekenen, maar juist hier hebben we een fout gemaakt. In plaats van sqrt hebben we per ongeluk sqr getypt, dat niet bestaat in de wiskunde module.
We probeerden dus te verwijzen naar een attribuut sqr dat niet bestaat en dat leidde tot de uitzondering AttributeError. De meesten van ons maken dit soort fouten vaak. U bent dus niet de enige.
Uitzonderingen afhandelen met Try Except
Als programmeur zullen de meesten van ons hun tijd besteden aan het schrijven van robuuste code die veerkrachtig is. Code die niet breekt door fouten. In Python kunnen we dit bereiken door onze verklaringen in een probeer - behalve verklaring.
Python Try-Except verklaring
Het try-except statement heeft de volgende structuur:
try: #uw code moet hier behalve """Geef hier uitzonderingstype(n) op"": #handel uitzondering hier af
Laten we de code insluiten in tracebackExp .py in een try-except verklaring.
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
Het uitvoeren van deze code levert de volgende uitvoer op
Dit is hoe het try-except statement werkt. Python voert de code in het try blok uit regel 7-8 Als er geen ongeldige code wordt gevonden, dan is de code in het except blok lijn 10 wordt overgeslagen en de uitvoering gaat door.
Maar als een ongeldige code wordt gevonden, stopt de uitvoering onmiddellijk in het try block en wordt gecontroleerd of de opgeworpen uitzondering overeenkomt met degene die we in het except statement hebben opgegeven regel 9 Als het overeenkomt, wordt de except-regel uitgevoerd en gaat het programma verder. Als het niet overeenkomt, wordt het programma onderbroken.
Het try-blok bevat meestal de code die een uitzondering kan oproepen, terwijl het except-blok de uitzondering opvangt en afhandelt.
Meerdere uitzonderingen behandelen met Except
We kunnen meerdere uitzonderingen afhandelen met een enkele "except" of meerdere "excepts". Het hangt allemaal af van hoe u elke uitzondering wilt afhandelen.
#1) Meerdere uitzonderingen behandelen met één uitzondering
try: #uw code moet hier komen except(Exception1[, Exception2[,...ExceptionN]]): #behandel uitzondering hier
Deze methode wordt gebruikt wanneer we vermoeden dat onze code verschillende uitzonderingen kan oproepen en we in elk geval dezelfde actie willen ondernemen. Dus, als de Python-interpreter een overeenkomst vindt, dan zal de code geschreven in het except-blok worden uitgevoerd.
Laten we de onderstaande Python-code als voorbeeld nemen
def get_fraction(value, idx): arr = [4,5,2,0] # een lijst met getallen idx_value = arr[idx] # als idx is> arr length, IndexError zal worden gegenereerd value/idx_value # als idx_value == 0, ZeroDivisionError zal worden gegenereerd if __name__ =='__main__': # stel 'value' en 'idx' in value = 54 idx = 3 # roep functie aan in een try-except statement. try: result = get_fraction(value, idx) print("Fraction is ", result) except(IndexError, ZeroDivisionError) als ex: print(ex)
Er zijn twee mogelijke uitzonderingen die hier aan de orde kunnen worden gesteld, ZeroDivisionError en IndexError Als een van deze uitzonderingen wordt opgeworpen, wordt het except-blok uitgevoerd.
In bovenstaande code is idx=3, dus idx_ waarde wordt 0 en waarde /idx_ waarde zal ZeroDivisionError opleveren
#2) Meerdere uitzonderingen behandelen met meerdere uitzonderingen
try: #uw code komt hier behalve Exception1: #handle exception1 hier behalve Exception2: #handle exception2 hier behalve ExceptionN: #handle exceptionN hier
Als we elke uitzondering afzonderlijk willen behandelen, dan kan dat op deze manier.
Beschouw de onderstaande voorbeeldcode in Python
def get_fraction(value, idx): arr = [4,5,2,0] # een lijst met getallen idx_value = arr[idx] # als idx is> arr length, IndexError zal worden gegenereerd value/idx_value # als idx_value == 0, ZeroDivisionError zal worden gegenereerd if __name__ =='__main__': # stel 'value' en 'idx' in value = 54 idx = 5 # roep functie aan in een try-excepts statement. try: result = get_fraction(value, idx) print("Fraction is ", result) exceptIndexError: print("idx van {} is buiten bereik".format(idx)) except ZeroDivisionError: print("arr[{}] is 0. Kan dus niet delen door nul".format(idx)) except Exception as ex: print(ex) print("Niet zeker wat er gebeurd is dus niet veilig om door te gaan, app wordt onderbroken") raise ex
We zien hier dat Exception werd gebruikt in het laatste except statement. Dat komt omdat het exception object Exception overeenkomt met elke exception. Daarom moet het altijd als laatste worden gebruikt, omdat Python stopt met het controleren van andere exception handlers zodra er één overeenkomt.
In de code hierboven, idx=5 dus arr[idx] zal verhogen IndexError omdat idx groter is dan de lengte van de lijst arr
Ook is het nooit veilig om door te gaan met de uitvoering als je niet zeker weet welke uitzondering door je toepassing is opgeworpen. Daarom hebben we het type Uitzondering om onverwachte uitzonderingen op te vangen. Dan informeren we de gebruiker en onderbreken we de toepassing door dezelfde uitzondering op te roepen.
Probeer Else verklaring
Dit is een optionele functie van exception handling en staat u toe code toe te voegen die u wilt uitvoeren als er geen fouten zijn opgetreden. Als er een fout optreedt, zal dit else-blok niet worden uitgevoerd.
Beschouw de onderstaande Python voorbeeldcode, open uw editor en sla de code op als elseTry.py
def fraction_of_one(divisor): value = 1/divisor # if divisor is zero, ZeroDivisionError will be raised return value if __name__ == '__main__': while True: try: # Get input from the user. # if input is not a valid argument for int(), ValueError will be raised divisor = int(input("Enter a divisor: ")) # call our function to compute the fraction value = fraction_of_one(divisor) except (ValueError,ZeroDivisionError): print("Invoer kan niet nul zijn en moet een geldige letterlijke zijn voor int(). Probeer het opnieuw!") anders: print("Waarde: ", waarde) break
We krijgen invoer van de gebruiker en gebruiken die om 1 te delen. We hebben hier twee mogelijke uitzonderingen, een ongeldige gebruikersinvoer die zal leiden tot ValueError en een nul(0) waardoor ZeroDivisionError Ons except statement handelt deze fouten af.
Nu willen we de waarde van waarde Ons else-blok zorgt ervoor dat het alleen wordt afgedrukt als ons try-blok zonder fout wordt uitgevoerd. Dit is belangrijk omdat als er een fout optreedt in ons try-blok, de waarde zal ongedefinieerd zijn. Dus, het openen ervan zal een andere fout opleveren.
Voer de bovenstaande code uit met Python elseTry.py
De uitvoer hierboven laat zien dat we voor de eerste invoer het volgende hebben getypt 0 en drukte op ENTER. Aangezien onze deler 0 kreeg, verhoogde 1/deler zeroDivisionError Onze tweede invoer was k, die ongeldig is voor int (), vandaar de uitzondering ValueError is verhoogd.
Maar onze laatste invoer was 9 die geldig is en als resultaat kregen we de waarde " waarde " afgedrukt als 0.11111111111111.
Probeer eindelijk Statement
Dit is ook een optionele functie van exception handling en zal altijd worden uitgevoerd, ongeacht wat er gebeurt in de exception handlers.
Dat wil zeggen:
- Of er al dan niet een uitzondering optreedt
- Zelfs als in de andere blokken een 'return' wordt aangeroepen.
- Zelfs als het script wordt gestopt in de andere blokken
Dus, als we een code hebben die we in alle situaties willen uitvoeren, is het finaal-blok onze man. Dit blok wordt meestal gebruikt voor opruimacties zoals het sluiten van bestanden.
Beschouw de onderstaande voorbeeldcode in Python
def readFile(file_path): try: openFile = open(file_path,'r') # Open een bestand als read-only print(openFile.readline()) # Lees de eerste regel van de bestandsinhoud except FileNotFoundError as ex: print(ex) finally: print("Cleaning...") openFile.close() if __name__ == '__main__': filePath = './text.txt' readFile(filePath)
Deze code probeert het bestand text.txt in de huidige directory te openen en te lezen. Als het bestand bestaat, drukt ons programma de eerste regel van het bestand af, waarna ons finally-blok wordt uitgevoerd en het bestand wordt gesloten.
Stel dat we een bestand genaamd text.txt hebben in de directory waar dit programmabestand staat en dat Hello bevat. Als we het programma uitvoeren, krijgen we de volgende uitvoer
Dit voorbeeld is met opzet gekozen omdat ik een klein probleem aan de orde wilde stellen dat kan optreden bij het sluiten van bestanden in het slot-blok.
Als het bestand niet bestaat, is de uitzondering FileNotFoundError wordt verhoogd en de variabele openFile zal niet gedefinieerd zijn en zal geen bestandsobject zijn. Daarom zal het proberen te sluiten in het slot-blok een uitzondering opleveren UnboundLocalError die een subklasse is van NameError .
Dit zegt in feite dat we proberen te verwijzen naar de variabele openFile voordat het is toegewezen.
Een kleine truc hier is het gebruik van exception handlers in het finally-blok.
def readFile(file_path): try: openFile = open(file_path,'r') # Open een bestand als read-only print(openFile.readline()) # Lees de eerste regel van de bestandsinhoud except FileNotFoundError as ex: print(ex) finally: try: print("Cleaning...") openFile.close() except: # vangt alle uitzonderingen op pas # Negeer deze fout omdat het ons niets kan schelen. if __name__ == '__main__': filePath = './text.txt' readFile(filePath)
Als ons try-block FileNotFoundError oplevert, dan krijgen we de volgende uitvoer
Verhoog Uitzondering
Het goede nieuws van Python excepties is dat we ze opzettelijk kunnen verhogen. Excepties worden verhoogd met de wervende verklaring .
De raise-instructie heeft de volgende syntaxis:
raise [ExceptionName[(*args: Object)]]
Open een terminal en haal een willekeurig uitzonderingsobject op uit de ingebouwde uitzonderingen van Python. Bijvoorbeeld, als we een ZeroDivisionError krijgen:
>>> raise ZeroDivisionError("Kan niet delen door nul")
We krijgen het spoor terug:
Waarom is het dan belangrijk om uitzonderingen te maken?
- Bij het werken met aangepaste uitzonderingen.
- Tijdens sanity checks.
Aangepaste Uitzonderingsklassen
Een aangepaste uitzondering is een uitzondering die u maakt om fouten af te handelen die specifiek zijn voor uw behoefte. De truc is dat we een klasse definiëren die is afgeleid van het object Uitzondering dan gebruiken we het raise statement om onze uitzonderingsklasse op te wekken.
Stel dat we de gebruikersinvoer willen controleren en er zeker van willen zijn dat de invoerwaarde niet negatief is (sanity check). Natuurlijk kunnen we de Python-uitzondering ValueError oproepen, maar we willen de fout aanpassen door hem een specifieke en zelfverklarende naam te geven, zoals InputIsNegativeError Maar deze uitzondering is geen in Python ingebouwde uitzondering.
Dus eerst maken we onze basisklasse die zal afleiden van Exception.
klasse CustomError(Exception): "Uitzondering van de basisklasse voor alle uitzonderingen van deze module" pass
Dan maken we onze uitzonderingsklasse die de basisklasse erft en onze specifieke fout afhandelt.
klasse InputIsNegativeError(CustomError): """Raised when User enters a negative value""" pass
Laten we dit testen
try: value = int(input()) if value <0: raise InputIsNegativeError # Raise exception if value is negative except InputIsNegativeError: # catch and handle exception print("Input value should't be negative")
De bovenstaande code vraagt om gebruikersinvoer, en controleert of die negatief is. Als dat zo is, roept het onze aangepaste uitzondering InputIsNegativeError op, die later wordt opgevangen in het except-statement.
Hieronder staat de volledige code:
klasse CustomError(Exception): "Uitzondering basisklasse voor alle uitzonderingen van deze module" 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("Invoerwaarde mag niet negatief zijn")
Als de invoerwaarde een negatief getal is, zoals -1, dan krijgen we de uitvoer:
Bekijk de Python doc voor meer details over Python aangepaste uitzonderingen.
Vaak gestelde vragen
V #1) Hoe handelt Python een uitzondering af?
Antwoord: Python handelt uitzonderingen af met de try-except verklaring De code die een uitzondering kan oproepen wordt geplaatst en uitgevoerd in de probeer blok terwijl de behalve blok bevat de code die de eventuele uitzonderingen afhandelt.
Vraag 2) Wat is een uitzondering maken in Python?
Antwoord: Wanneer de Python-interpreter een ongeldige code tegenkomt, roept hij een exception op, wat Python's eigen manier is om ons te vertellen dat er iets onverwachts is gebeurd. We kunnen ook opzettelijk excepties oproepen met de wervende verklaring .
V #3) Hoe handelt Python meerdere uitzonderingen af?
Antwoord: Python behandelt meerdere uitzonderingen door middel van een enkel except-blok of meerdere except-blokken.
Voor een enkel blok worden de uitzonderingen doorgegeven als een tupel: behalve (Uitzondering1, Uitzondering2,...,UitzonderingN) en Python controleert van rechts naar links op een overeenkomst. In dit geval wordt voor elke uitzondering dezelfde actie ondernomen.
Een andere manier om alle uitzonderingen te vangen is door de naam van de uitzondering na het except keyword weg te laten.
except: # behandel hier alle uitzonderingen
De tweede manier is het gebruik van een except blok voor elke uitzondering:
behalve Uitzondering1: # code om Uitzondering1 af te handelen moet hier komen behalve Uitzondering2: # code om Uitzondering2 af te handelen moet hier komen behalve UitzonderingN: # code om UitzonderingN af te handelen moet hier komen
Zo kunt u voor elke uitzondering afzonderlijke acties ondernemen.
V #4) Waarom is Exception handling belangrijk in Python?
Antwoord: Het voordeel van het afhandelen van uitzonderingen in Python is dat we robuuste, schone en foutloze toepassingen kunnen maken. We willen niet dat onze productiecode crasht door enkele fouten, dus we handelen de fouten af en houden onze toepassing draaiende.
V #5) Hoe negeer je een uitzondering in Python?
Antwoord: Om een uitzondering in Python te negeren, gebruik je de pas sleutelwoord in het except blok. Laten we zeggen dat we de ValueError uitzondering willen negeren. Dat doen we op deze manier:
behalve ValueError: pass
Tenzij u weet wat u doet, is het een slechte gewoonte om uitzonderingen te negeren. Informeer de gebruiker tenminste over alle mogelijke fouten.
Conclusie
In deze tutorial hebben we behandeld: Python Uitzonderingen, Traceback; hoe omgaan met uitzonderingen met Probeer / Behalve / Anders / Eindelijk blokken, hoe Verhoog Uitzonderingen, en tenslotte hoe we onze eigen aangepaste uitzonderingen kunnen maken.
Bedankt voor het lezen!