Python Try Except - Python-ek salbuespena kudeatzen adibideekin

Tutorial honek Pythonen Salbuespenak kudeatzea azaltzen du Try Except blokea erabiliz, programazio-adibideen laguntzarekin:

Bi errore motak Python programa bat bat-batean gelditzea eragin dezakete, hau da, Sintaxia. Erroreak eta Salbuespenak . Tutorial honetan, bigarren errore mota (salbuespenak) eztabaidatuko dugu hainbat gai garrantzitsuren azpian.

Asko onuragarria izango zaigu gure aplikazioan salbuespenak kudeatzeaz, hala nola:

  • Aplikazio sendo bat sortzea.
  • Kode garbi eta errorerik gabeko bat sortzea.

Python Try Except

Albiste on bat da Python-ek salbuespen kopuru ona duela barneratuta gure kodean akatsak atzemateko. Gainera, salbuespen pertsonalizatuak sortzeko aukera ematen digu barneratutako salbuespenetako bat ere gure beharretara egokitzen ez denean.

Zer da Salbuespen bat

Beraz, zer da salbuespen bat Python-en? Beno, modu sinplean esanda, Python interpreteak kode baliogabea exekutatzen saiatzen den bakoitzean, salbuespen bat sortzen du, eta salbuespen hori kudeatzen ez den kasuetan, programaren jarraibideen fluxu normala eten egiten du eta traza bat inprimatzen du.

Sor dezagun baliogabeko kode bat eta ikus dezagun nola erantzungo duen Python interpreteak.

Ireki Python shell bat eta exekutatu hurrengo kodea.

>>> 50/0

Hau da hauetako bat. programazioaren akats ohikoenak. Goiko kodea 50 zenbakia 0 (zero) zatitzen saiatzen da. Python openFile aldagaia esleitu baino lehen.

Hemen trikimailu txiki bat azkenly-blokearen barruan salbuespen-kudeatzaileak erabiltzea da.

def readFile(file_path): try: openFile = open(file_path,'r') # Open a file as read-only print(openFile.readline()) # Read first line of file content except FileNotFoundError as ex: print(ex) finally: try: print("Cleaning...") openFile.close() except: # catches all exceptions pass # Ignore this error because we don't care. if __name__ == '__main__': filePath = './text.txt' readFile(filePath) 

Gure try-blokeak FileNotFoundError sortzen badu, hurrengo irteera izango dugu

Sortu Salbuespena

Pythonen salbuespenei buruzko albiste on bat nahita egin dezakegula da. igo itzazu. Salbuespenak raise statement rekin sortzen dira.

Graise instrukzioak honako sintaxia du:

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

Ireki terminal bat eta edozein salbuespen objektu planteatu. Python-en barneko Salbuespenak. Adibidez, ZeroDivisionError planteatzen badugu:

>>> raise ZeroDivisionError("Can't divide by zero")

Trazapena lortuko dugu:

Beraz, zergatik da garrantzitsua salbuespenak planteatzea?

  • Salbuespen pertsonalizatuekin lan egitean.
  • Zentzuzko egiaztapenetan.

Salbuespen klase pertsonalizatuak

Salbuespen pertsonalizatua zure beharretarako espezifikoak diren erroreak kudeatzeko sortzen duzuna da. Trikimailua da Salbuespena objektutik eratortzen den klase bat definitzen dugula, eta, ondoren, planteatu adierazpena erabiltzen dugula gure salbuespen klasea igotzeko.

Demagun erabiltzailearen sarrera egiaztatu nahi dugula eta ziurtatu sarrerako balioa ez da negatiboa (sanity check). Noski, Python-en salbuespena ValueError plantea genezake, baina errorea pertsonalizatzea gustatuko litzaiguke InputIsNegativeError bezalako izen espezifiko eta zehatza emanez. Baina salbuespen hau ez da Python-en barnean dagoenaSalbuespena.

Beraz, lehenik eta behin, Salbuespenetik aterako den gure oinarrizko klasea sortzen dugu.

class CustomError(Exception): "Base class exception for all exceptions of this module" pass 

Ondoren, oinarrizko klasea heredatuko duen eta gure errore espezifikoa kudeatuko duen gure salbuespen klasea sortzen dugu.

class InputIsNegativeError(CustomError): """Raised when User enters a negative value""" pass 

Proba dezagun hau

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") 

Erabiltzaileak sartzeko goiko kodea eskaera, eta egiaztatu negatiboa den. Egia bada, gure salbuespen pertsonalizatua sortzen du InputIsNegativeError, geroago except-adierazpenean atzematen dena.

Behean kode osoa dago:

class CustomError(Exception): "Base class exception for all exceptions of this 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 exception print("Input value shouldn't be negative") 

Sarrerako balioa bada. -1 bezalako zenbaki negatiboa da, orduan irteera izango dugu:

Begiratu Python dokumentua Python salbuespen pertsonalizatuei buruzko xehetasun gehiago lortzeko.

Maiz egiten diren galderak

G #1) Nola kudeatzen du Python-ek salbuespen bat?

Erantzuna: Python-ek salbuespenak kudeatzen ditu erabiliz. saiatu-salbu adierazpena . Salbuespen bat sor dezakeen kodea saiatu blokean jarri eta exekutatzen da, eta salbuespen blokeak k salbuespenak kudeatuko dituen kodea gordetzen du, halakorik gertatuz gero.

G #2) Zer da salbuespen bat piztea Python-en?

Erantzuna: Python-en interpreteak baliogabeko kode bat aurkitzen duen bakoitzean, salbuespen bat sortzen du, Python-en berezko modua. ustekabeko zerbait gertatu zela esateko. Nahita ere salbuespenak sor ditzakegu goratu adierazpena erabiliz.

G #3) Nola kudeatzen ditu Pythonek salbuespen anitz?

Erantzuna: Python-ek hainbat salbuespen kudeatzen dituBloke bakar bat izan ezik edo bloke bat baino gehiago izan ezik.

Bloke bakarrerako, salbuespenak tupla gisa pasatzen dira: salbuespen (Salbuespena1, Salbuespena2,..,ExceptionN) eta Python egiaztapenak eskuinetik ezkerrerako partida baterako. Kasu honetan, ekintza bera egiten da salbuespen bakoitzerako.

Salbuespen guztiak harrapatzeko beste modu bat salbuespenaren izena salbuespen gako-hitzaren ondoren kanpoan uztea da.

except: # handle all exceptions here

Bigarren modua da. salbuespen bakoitzeko salbuespen bloke bat erabiltzeko:

except Exception1: # code to handle Exception1 goes here except Exception2: # code to handle Exception2 goes here except ExceptionN: # code to handle ExceptionN goes here 

Horrela, ekintza bereiziak egin ditzakezu Salbuespen bakoitzeko.

G #4) Zergatik da garrantzitsua Salbuespenen kudeaketa Python-en?

Erantzuna: Python-en salbuespenak kudeatzeko onura aplikazio sendoak, garbiak eta errorerik gabekoak sor ditzakegula da. Ez dugu nahi gure produkzio-kodea hutsegitea errore batzuen ondorioz, beraz, akatsak kudeatzen ditugu eta gure aplikazioa martxan mantentzen dugu.

G #5) Nola ez ikusi salbuespen bat Python-en?

Erantzuna: Python-en salbuespen bat baztertzeko, erabili pass gako-hitza except blokean. Demagun ValueError salbuespena alde batera utzi nahi dugula. Honela egingo dugu:

except ValueError: pass

Zer egiten ari zaren jakin ezean, praktika txarra da salbuespenak alde batera uztea. Gutxienez, jakinarazi erabiltzaileari balizko akats guztien berri.

Ondorioa

Tutorial honetan, hauek landu ditugu: Python Exceptions, Traceback; nola kudeatu salbuespenak Saiatu / Salbu / Bestela / Azkenik blokeak, nola Gratu Salbuespenak eta, azkenik, gure Salbuespen pertsonalizatuak nola sortu.

Eskerrik asko irakurtzeagatik!

interpreteak eragiketa baliogabe gisa ikusten du eta ZeroDivisionErrorbat sortzen du, programa eten egiten du eta aztarna bat inprimatzen du.

Argi ikus dezakegu hori. ZeroDivisionError planteatutako salbuespena da. Izan ere, Python-en modua da zenbaki bat zeroz zatitzea ez dela ona esateko. JavaScript bezalako beste hizkuntza batzuetan bada ere, hau ez da errore bat; eta python-ek zorrozki debekatzen du praktika hau.

Gainera, garrantzitsua da jakitea hori salbuespen objektu bat besterik ez dela eta Python-ek horrelako objektu asko barneratuta dituela. Begiratu Python-en dokumentazio ofizial hau Python-en barne-salbuespen guztiak ikusteko.

Traceback ulertzea

Salbuespenak kudeatzen hasi aurretik, salbuespenak gertatzen badira zer gertatuko den ulertzen lagunduko duela uste dut. ez dira kudeatzen eta nola egiten duen Python-ek gure errorearen berri emateko.

Python-ek erroreren bat aurkitzen duen bakoitzean, salbuespen bat sortzen du. Salbuespen hau kudeatzen ez bada, Traceback izeneko informazioa sortzen du. Beraz, zer informazio dauka aztarnatze honek?

Hona dauka:

  • Zer salbuespen sortu den eta salbuespen hori izan baino lehen zer gertatu den esaten digun errore-mezua. planteatu.
  • Errore hau eragin duten kodearen hainbat lerro-zenbakiak. Errore bat deien pila deitzen den funtzio-deien sekuentzia batek eragin dezake gero hemen eztabaidatuko duguna.

Nahiz eta bat den.pixka bat nahasia, hurrengo adibideak gure ulermenari argi gehiago emango diola agintzen dugu.

Gogoan 50 0 zatitik inprimatutako aztarnak gogoratu, atzemateak informazio hau duela ikus dezakegu:

  • Fitxategia “”: Honek kode hau kontsolako terminal batetik exekutatu dela adierazten digu.
  • 1. lerroa: Akatsa lerro zenbaki honetan gertatu dela adierazten digu.
  • ZeroDivisionError: zatiketa zero: Zein salbuespen sortu den eta zerk eragin duen esaten digu.

Saia gaitezen beste adibide bat eta agian ikusi deien pila nolakoa den. Ireki editore bat, idatzi beheko kodea eta gorde 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 

Ireki fitxategi hau aurkitu eta exekutatu den direktorioko terminal bat.

python tracebackExp.py

Ondoko aztarnak ikusiko dituzu:

Goiko aztarnak nahasia dirudi baina benetan, ez da. Pythonistek traza irakurtzeko modurik onena asmatu zuten, hau da, behetik gora . Beraz, erabil dezagun jakinduria hau aztarna honek zer eskaintzen duen ulertzen saiatzeko.

  • Behealdean, planteatu den salbuespena eta zergatik planteatu den aurkituko dugu.
  • Gora mugituz, tracebackExp .py fitxategi-izena jasoko dugu non errore hau gertatu den, errore hau eragin duen kalkulua compute = numb/div, funtzio pila 2 eta kalkulu hau egin den esteka-zenbaki-lerroa 6. .
  • Gorantz mugituz, gure stack2 funtzioa ikusten dugu3. lerroko pila 1 funtzioan deitzen zen.
  • Goikora mugituz, pila 1 funtzioari 11. zenbakian deitzen zaiola ikusiko dugu. modulua > exekutatzen ari den fitxategia dela esaten digu.

Python-en salbuespen arruntak

Python liburutegiak integratutako salbuespen asko definitzen ditu. Python-en dokumentazioa egiaztatu dezakezu edo barneko local () funtziora deitu dezakezu behean bezala:

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

Ez gara salbuespen guzti hauek zuzentzen saiatuko, baina ohiko salbuespen batzuk ikusiko ditugu. litekeena da topatuko duzula.

#1) TypeError

Eragiketa edo funtzio bat mota desegokiko objektu bati aplikatzen zaionean sortzen da.

1. adibidea

Kontuan izan beheko programa. Dibidendua eta zatitzailea hartzen ditu, gero dibidendua zatitzailearekin zatitzearen emaitza kalkulatzen eta inprimatzen du.

def compute_division(): dividend = int(input("Enter the dividend: ")) # cast string to int divisor = input("Enter the divisor: ") # no casting # Compute division result = dividend/divisor # print result print("The result of {}/{} is: {}".format(dividend, divisor, result)) if __name__ == '__main__': result = compute_division() 

Dibidenduaren eta zatitzailearen balioa eskatzen diogu erabiltzaileari, baina zatitzailearen katea botatzea ahaztu egiten zaigu. balioa zenbaki oso batean. Beraz, amaitzen dugu dibidenduaren mota integer( int ) eta zatitzailearen mota string( str ). Ondoren, TypeError jasoko dugu zatiketa-operadoreak (/) ez duelako kateetan funtzionatzen.

Python-ek ez bezala jakitea interesgarria izan daiteke. Javascript-ek Type Coercion du, funtsean, eragigaien mota bat beste eragiketa motaren balio baliokide batera bihurtzen du eragigaiak direnean.mota desberdinak.

#2) BalioErrorea

Hau sortzen da eragiketa edo funtzio batek mota egokia baina balio desegokia duen argumentua jasotzen duenean.

Adibidea 2

Kontuan izan goiko 1.adibidea ko gure programa.

Erabiltzaileak '3a' bezalako dibidendurako balio alfanumeriko bat sartzen badu, orduan gure programak gora egingo du. ValueError salbuespena. Hau da, Python int() metodoak edozein zenbaki edo kate hartzen duen eta osoko objektu bat itzultzen duen arren, katearen balioak ez luke letrarik edo zenbakizko baliorik izan behar.

#3) AttributeError

Salbuespen hau existitzen ez den atributu bat esleitzean edo erreferentzia egitean sortzen da.

3. adibidea

Kontuan izan programa. behean. Zenbaki bat hartzen du eta bere erro karratua kalkulatzen du Python matematika modulua erabiliz

import math # import math library to gain access to its code def compute_square_root(number): # compute the square root using the 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 

Erabiltzaile batek zenbaki bat sartzen duenean, gure programa matematikako moduluko funtzio bat erabiltzen saiatzen da bere erro karratua kalkulatzeko, baina hori hemen, akats bat egin dugu. Sqrt-en ordez, oker idatzi dugu matematikako moduluan existitzen ez den sqr.

Beraz, existitzen ez den sqr atributu bati erreferentzia egiten saiatzen ari ginen eta gidatu genuen. AttributeError sortzen ari den salbuespenari. Gutako gehienok horrelako akatsak asko egiten ditugu. Beraz, ez zaude bakarrik.

Try Except-ekin salbuespenak kudeatzea

Programatzaile gisa, gehienok denbora pasako dugun gauza bat kode sendo bat idaztea da.erresistenteak. Akats batzuengatik apurtzen ez den kodea. Python-en, gure adierazpenak saiatu salbu adierazpen baten barruan sartuz lor dezakegu.

Python Try-Except adierazpena

Try-except adierazpenak egitura hau du:

try: #your code goes here except """Specify exception type(s) here""": #handle exception here 

Jar dezagun kodea tracebackExp .py-n try-except adierazpen baten barruan.

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 

Kode hau exekutatzeak irteera emango du

Horrela funtzionatzen du saiatu-salbu adierazpenak. Python-ek try blokean exekutatzen du kodea 7-8 lerroa . Kode baliogaberik aurkitzen ez bada, 10. lerroko salbuespen blokeko kodea saltatu egingo da eta exekuzioak jarraitzen du.

Baina, kode baliogabe bat aurkitzen bada, exekuzioa berehala geldituko da. saiatu blokeatu eta egiaztatzen du planteatutako salbuespena bat datorren 9. lerroa except adierazpenean emandakoarekin. Bat badator, salbu blokea exekutatu eta jarraitzen du. Hala egiten ez badu, programak eten egingo du.

Try-blokeak normalean salbuespen bat sor dezakeen kodea dauka, except-blokeak salbuespena harrapatzen eta kudeatzen duen bitartean.

Anitzak kudeatzea Salbuespenak Salbuespenarekin

Salbuespen anitz kudea ditzakegu "salbu" bakarrarekin edo "salbuespen" anitzekin. Salbuespen bakoitza nola kudeatu nahi duzunaren araberakoa da dena.

#1) Salbuespen anitzak kudeatzea Salbuespen bakarrarekin

try: #your code goes here except(Exception1[, Exception2[,...ExceptionN]]]): #handle exception here 

Metodo hau gure kodea izan daitekeela susmatzen dugunean erabiltzen da.salbuespen desberdinak planteatu eta kasu bakoitzean ekintza bera egin nahi dugu. Beraz, Python interpreteak bat-etortze bat aurkitzen badu, salbu blokean idatzitako kodea exekutatuko da.

Kontuan izan dezagun beheko Python kodea adibidea

def get_fraction(value, idx): arr = [4,5,2,0] # a list of numbers idx_value = arr[idx] # if idx is > arr length, IndexError will be raised value/idx_value # if idx_value == 0, ZeroDivisionError will be raised 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) 

Bi ditugu. Hemen plantea daitezkeen salbuespen posibleak, ZeroDivisionError eta IndexError . Salbuespen hauetakoren bat planteatzen bada, except blokea exekutatuko da.

Goiko kodean, idx=3, beraz, idx_ balioa 0 bihurtzen da eta balioa /idx_ value ZeroDivisionError sortuko du

#2) Salbuespen anitzak kudeatzea Salbuespen anitzekin

try: #your code goes here except Exception1: #handle exception1 here except Exception2: #handle exception2 here except ExceptionN: #handle exceptionN here 

Nahiago badugu kudeatu salbuespen bakoitza bereizita, horrela egin dezakezu.

Kontuan izan beheko Python kodearen adibidea

def get_fraction(value, idx): arr = [4,5,2,0] # a list of numbers idx_value = arr[idx] # if idx is > arr length, IndexError will be raised value/idx_value # if idx_value == 0, ZeroDivisionError will be raised 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) except IndexError: print("idx of {} is out of range".format(idx)) except ZeroDivisionError: print("arr[{}] is 0. Hence, can't divide by zero".format(idx)) except Exception as ex: print(ex) print("Not sure what happened so not safe to continue, \ app will be interrupted") raise ex 

Hemen konturatu gara Exception azken except adierazpenean erabili zela. . Hau da, salbuespen objektua Exception edozein salbuespen bat datorrelako. Horregatik, beti azkena izan behar du, Python-ek beste salbuespen-kudeatzaile batzuk egiaztatzeari utziko baitio bat datorrenean.

Goiko kodean, idx=5 , beraz, arr[idx ] -k IndexError igoko du, idx zerrendaren luzera baino handiagoa delako arr

Gainera, zure aplikazioak zer salbuespen planteatu duen ziur ez da inoiz segurua exekutatzen jarraitzeko. Horregatik, Salbuespen mota dugu aurreikusi gabeko salbuespenak harrapatzeko. Ondoren, jakinaraziko dioguerabiltzailea eta eten aplikazioa salbuespen bera planteatuz.

Saiatu Else adierazpena

Hau aukerazko funtzioa da salbuespenak kudeatzeko eta nahi duzun kodea gehitzeko aukera ematen dizu. exekutatu errorerik gertatu ez denean. Errore bat gertatzen bada, beste bloke hau ez da exekutatuko.

Kontuan hartu beheko Python kodea adibidea, ireki zure editorea eta gorde kodea 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("Input can't be zero and should be a valid literal for int(). Please, try again!") else: print("Value: ", value) break 
0 gisa>Erabiltzailearen sarrera jasotzen dugu eta 1 zatitzeko erabiltzen dugu. Bi salbuespen posible ditugu hemen, erabiltzailearen sarrera baliogabe bat ValueErroreragingo duena eta zero(0)eragingo duena. ZeroDivisionError. Gure except adierazpenak errore hauek kudeatzen ditu.

Orain, balioa ren balioa inprimatu nahi dugu. Gure else-blokeak ziurtatzen du inprimatuta dagoela gure try blokea errorerik gabe exekutatzen bada. Hau garrantzitsua da gure try-blokean errore bat gertatzen bada, balioa definitu gabe egongo delako. Beraz, hara sartzeak beste errore bat sortuko du.

Exekutatu goiko kodea Python elseTry.py-rekin

Goiko irteerak erakusten du hori lehen sarrerarako, 0 idatzi eta ENTER sakatu dugu. Gure zatitzaileak 0 jaso zuenez, 1/zatitzaileak zeroDivisionError planteatu zuen. Gure bigarren sarrera k izan zen, int (rentzako baliogabea dena), beraz, ValueError salbuespena planteatzen da.

Baina gure azken sarrera 9 izan zen, balio duena eta gisa. emaitza, " balioa "-ren balioa 0.1111111111111111

Saiatu Azkenean inprimatuta lortu dugu.Adierazpena

Hau ere salbuespenak kudeatzeko aukerako funtzioa da eta beti exekutatuko da salbuespenen kudeatzaileetan gertatzen dena gorabehera.

Hau da:

  • Salbuespena gertatzen den ala ez
  • Beste blokeetan 'itzulera' deitzen bada ere.
  • Beste blokeetan scripta irten bada ere

Beraz, egoera guztietan exekutatu nahi dugun kode bat badugu, azkenik-blokea da gure mutila. Bloke hau fitxategiak ixteko bezalako garbiketarako erabiltzen da gehienbat.

Kontuan izan beheko Python kodea adibidea

def readFile(file_path): try: openFile = open(file_path,'r') # Open a file as read-only print(openFile.readline()) # Read first line of file content except FileNotFoundError as ex: print(ex) finally: print("Cleaning...") openFile.close() if __name__ == '__main__': filePath = './text.txt' readFile(filePath) 

Kode hau text.txt fitxategia ireki eta irakurtzen saiatzen da. bere egungo direktorioan. Fitxategia existitzen bada, orduan gure programak fitxategiaren lehen lerroa inprimatuko du eta, ondoren, gure finally-blokea exekutatuko da eta fitxategia itxiko da.

Esan programaren fitxategi honen direktorioan text.txt izeneko fitxategi bat dugula. Kaixo da eta dauka. Programa exekutatzen badugu, irteera izango dugu

Adibide hau nahita aukeratu da, azkenean fitxategiak ixtean gerta daitekeen arazo txiki bat konpontzea nahi nuelako. blokea.

Fitxategia existitzen ez bada, FileNotFoundError salbuespena sortuko da eta openFile aldagaia ez da definituko eta ez da fitxategi bat izango. objektua. Hori dela eta, finally-blokean ixten saiatzeak UnboundLocalError salbuespena sortuko du, hau da, NameError azpiklase bat dena.

Honek, funtsean, erreferentzia egiten saiatzen ari garela esaten du. du

Gora joan