Bu eğitim, Python'da Try Except bloğunu kullanarak Exception Handling'i programlama örnekleri yardımıyla açıklamaktadır:
İki hata türü bir Python programının aniden durmasına neden olabilir, örn. Sözdizimi Hataları ve İstisnalar Bu eğitimde, ikinci hata türünü (İstisnalar) birkaç önemli başlık altında ele alacağız.
Uygulamamızdaki istisnaları ele almaktan çok faydalanacağız:
- Sağlam bir uygulama oluşturma.
- Temiz ve hatasız bir kod oluşturmak.
Python Try Except
İyi haberlerden biri, Python'un kodumuzdaki hataları yakalamak için çok sayıda yerleşik istisnaya sahip olmasıdır. Ayrıca, yerleşik istisnalardan hiçbiri ihtiyaçlarımıza uymadığında bize özel istisnalar oluşturma fırsatı verir.
İstisna Nedir
Peki Python'da istisna nedir? Basit bir ifadeyle, Python yorumlayıcısı geçersiz bir kodu çalıştırmaya çalıştığında, bir istisna oluşturur ve böyle bir istisnanın işlenmediği durumlarda, programın talimatlarının normal akışını bozar ve bir geri izleme yazdırır.
Geçersiz bir kod oluşturalım ve Python yorumlayıcısının nasıl tepki vereceğini görelim.
Bir Python kabuğu açın ve aşağıdaki kodu çalıştırın.
>>> 50/0
Bu, programlamada en yaygın hatalardan biridir. Yukarıdaki kod, sayıyı bölmeye çalışır 50 tarafından 0 (sıfır). Python yorumlayıcısı bunu geçersiz bir işlem olarak görür ve bir ZeroDivisionError , programı bozar ve bir geri izleme yazdırır.
Şunu açıkça görebiliriz ZeroDivisionError Bu aslında Python'un bize bir sayıyı sıfıra bölmenin hoş olmadığını söylemenin kendi yoludur. JavaScript gibi diğer dillerde bu bir hata değildir; ve python bu uygulamayı kesinlikle yasaklar.
Ayrıca, bunun sadece bir istisna nesnesi olduğunu ve Python'un yerleşik olarak bu tür birçok nesneye sahip olduğunu bilmek önemlidir. Tüm Python Yerleşik İstisnalarını görmek için bu Python resmi belgesine göz atın.
Traceback'i Anlama
İstisnaları ele almaya başlamadan önce, istisnalar ele alınmazsa tam olarak ne olacağını ve Python'un hatamız hakkında bizi bilgilendirmek için elinden geleni nasıl yaptığını anlamanın yardımcı olacağını düşünüyorum.
Python bir hata ile karşılaştığında bir istisna üretir. Eğer bu istisna işlenmezse, Traceback adı verilen bir bilgi üretir. Peki, bu traceback hangi bilgileri içerir?
İçerisinde:
- Hangi istisnanın ortaya çıktığını ve bu istisna ortaya çıkmadan önce neler olduğunu anlatan hata mesajı.
- Bu hataya neden olan kodun çeşitli satır numaraları. Bir hataya, bir dizi işlev çağrısı neden olabilir. çağrı yığını Bunu daha sonra burada tartışacağız.
Biraz kafa karıştırıcı olsa da, bir sonraki örneğin anlayışımıza daha fazla ışık tutacağına söz veriyoruz.
Yukarıda 50'yi 0'a böldüğümüzde yazdırılan geri bildirimi hatırlarsak, geri bildirimin aşağıdaki bilgileri içerdiğini görebiliriz:
- Dosya "": Bu bize bu kodun bir konsol terminalinden çalıştırıldığını gösterir.
- satır 1: Bu bize hatanın bu satır numarasında meydana geldiğini söyler.
- ZeroDivisionError: tarafından bölünme Sıfır: Bize hangi istisnanın ortaya çıktığını ve buna neyin sebep olduğunu söyler.
Başka bir örnek deneyelim ve belki de bir çağrı yığını Bir editör açın, aşağıdaki kodu girin ve şu şekilde kaydedin 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
Bu dosyanın bulunduğu dizinde bir terminal açın ve çalıştırın.
python tracebackExp.py
Aşağıdaki geri bildirimi göreceksiniz:
Yukarıdaki traceback kafa karıştırıcı görünebilir ama aslında öyle değil. Pythonistalar traceback'i okumanın en iyi yolunu buldular, bu da aşağıdan yukarıya Şimdi bu bilgeliği kullanarak bu geri izlemenin neler sunabileceğini anlamaya çalışalım.
- En altta, ortaya çıkan istisnayı ve neden ortaya çıktığını öğreniriz.
- Yukarı çıkarsak, dosya adını alırız tracebackExp .py bu hatanın oluştuğu yer, bu hataya neden olan hesaplama compute = numb/div, stack2 işlevi ve bu hesaplamanın gerçekleştirildiği 6 numaralı bağlantı satırı.
- Yukarı çıktığımızda, stack2 fonksiyonumuzun 3 numaralı satırda stack1 fonksiyonu içinde çağrıldığını görüyoruz.
- En üste gittiğimizde, stack1 fonksiyonunun 11 numaralı satırda çağrıldığını görüyoruz. < modül > bize çalıştırılanın dosya olduğunu söyler.
Yaygın Python İstisnaları
Python kütüphanesi çok sayıda yerleşik istisna tanımlar. Python Dokümantasyonunu kontrol edebilir veya yerleşik yerel () fonksiyonunu aşağıdaki gibi kullanın:
>>> dir(locals()['__builtins__'])
Tüm bu istisnaları ele almaya çalışmayacağız, ancak muhtemelen karşılaşacağınız birkaç yaygın istisnayı göreceğiz.
#1) TypeError
Bir işlem veya fonksiyon uygun olmayan türdeki bir nesneye uygulandığında ortaya çıkar.
Örnek 1
Aşağıdaki programı ele alalım. Bu program bir bölen ve bölünen alır, ardından bölenin bölünene bölünmesinin sonucunu hesaplar ve yazdırır.
def compute_division(): dividend = int(input("Enter the dividend: ")) # cast string to int divisor = input("Enter the divisor: ") # no casting # Bölme işlemini hesapla result = dividend/divisor # print result print("The result of {}/{} is: {}".format(dividend, divisor, result)) if __name__ == '__main__': result = compute_division()
Kullanıcıdan bölen ve bölenin değerini talep ediyoruz, ancak bölenin string değerini bir tamsayıya dönüştürmeyi unutuyoruz. Bu nedenle, bölenin türünün tamsayı( int ) ve bölenin türünün string( str ). Daha sonra TipHatası bölme operatörü (/) dizeler üzerinde işlem yapmadığından.
Python'dan farklı olarak Javascript'in, işlenenler farklı türlerde olduğunda işlenenin türlerinden birini diğer işlenenin türünün eşdeğer bir değerine dönüştüren Tip Zorlama özelliğine sahip olduğunu bilmek ilginizi çekebilir.
#2) ValueError
Bir işlem veya fonksiyon doğru türde ancak uygun olmayan bir değere sahip bir argüman aldığında bu durum ortaya çıkar.
Örnek 2
Programımızı düşünün Örnek 1 Yukarıda.
Kullanıcı kar payı için '3a' gibi alfanümerik bir değer girerse, programımız ValueError istisnasını yükseltir. Bunun nedeni, Python int() yönteminin herhangi bir sayı veya dizeyi alıp bir tamsayı nesnesi döndürmesine rağmen, dize değerinin harf veya sayısal olmayan herhangi bir değer içermemesi gerektiğidir.
#3) AttributeError
Bu istisna, mevcut olmayan bir öznitelik atanırken veya referans verilirken ortaya çıkar.
Örnek 3
Aşağıdaki programı düşünün. Bir sayı alır ve Python matematik modülünü kullanarak karekökünü hesaplar
import math # koduna erişmek için math kütüphanesini içe aktar def compute_square_root(number): # math kütüphanesini kullanarak karekökü hesapla result = math.sqr(number) return result if __name__ == '__main__': # kullanıcıdan hesaplama için girdi al number = int(input("Compute Square root of: ")) # karekökü hesaplamak için fonksiyonu çağır
Kullanıcı bir sayı girdiğinde, programımız bu sayının karekökünü hesaplamak için matematik modülündeki bir fonksiyonu kullanmaya çalışır, ancak burada bir hata yaptık. sqrt yerine, yanlışlıkla matematik modülünde bulunmayan sqr yazdık.
Yani, var olmayan bir sqr özniteliğine başvurmaya çalışıyorduk ve AttributeError istisnasının ortaya çıkmasına neden olduk. Çoğumuz bu tür hataları çok yaparız, yani yalnız değilsiniz.
Try Except ile İstisnaları İşleme
Bir programcı olarak, çoğumuzun zamanımızı harcayacağı bir şey, esnek olan sağlam bir kod yazmaktır. Bazı hatalar nedeniyle bozulmayan kod. Python'da bunu, ifadelerimizi bir dene - hariç Açıklama.
Python Try-Except deyimi
try-except deyimi aşağıdaki yapıya sahiptir:
try: #your code goes here except """Specify exception type(s) here""": #handle exception here
Kodu içine alalım tracebackExp .py içinde bir try-except deyimi.
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 sürekli") # 15
Bu kodu çalıştırmak şu çıktıyı üretecektir
try-except deyimi şu şekilde çalışır: Python try bloğundaki kodu çalıştırır 7-8. satır Geçersiz bir kod bulunmazsa, except bloğundaki kod 10. satır atlanır ve yürütme devam eder.
Ancak, geçersiz bir kod bulunursa, yürütme try bloğunda hemen durur ve yükseltilen istisnanın except deyiminde sağladığımızla eşleşip eşleşmediğini kontrol eder 9. satır Eşleşirse, except bloğu çalıştırılır ve devam eder. Eşleşmezse, program kesilir.
try bloğu genellikle bir istisna oluşturabilecek kodu içerirken, except bloğu istisnayı yakalar ve işler.
Except ile Birden Fazla İstisnayı İşleme
Birden fazla istisnayı tek bir "except" veya birden fazla "excepts" ile ele alabiliriz. Bu tamamen her bir istisnayı nasıl ele almak istediğinize bağlıdır.
#1) Tek Bir Except ile Birden Fazla İstisnayı İşleme
try: #your code goes here except(Exception1[, Exception2[,...ExceptionN]]): #handle exception here
Bu yöntem, kodumuzun farklı istisnalar oluşturabileceğinden şüphelendiğimizde ve her durumda aynı işlemi yapmak istediğimizde kullanılır. Böylece, Python yorumlayıcısı bir eşleşme bulursa, except bloğunda yazılan kod yürütülür.
Aşağıdaki örnek Python kodunu ele alalım
def get_fraction(value, idx): arr = [4,5,2,0] # bir sayı listesi idx_value = arr[idx] # idx> arr uzunluğu ise, IndexError yükseltilecektir value/idx_value # idx_value == 0 ise, ZeroDivisionError yükseltilecektir 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)
Burada gündeme getirilebilecek iki olası istisna bulunmaktadır, ZeroDivisionError ve IndexError Bu istisnalardan herhangi biri ortaya çıkarsa, except bloğu yürütülür.
Yukarıdaki kodda idx=3, yani idx_ değer 0 olur ve değer /idx_ değer ZeroDivisionError yükseltecektir
#2) Çoklu Excepts ile Çoklu İstisnaları İşleme
try: #your code goes here except Exception1: #handle exception1 here except Exception2: #handle exception2 here except ExceptionN: #handle exceptionN here
Her bir istisnayı ayrı ayrı ele almak istersek, bunu şu şekilde yapabilirsiniz.
Aşağıdaki örnek Python kodunu ele alalım
def get_fraction(value, idx): arr = [4,5,2,0] # bir sayı listesi idx_value = arr[idx] # idx> arr uzunluğu ise, IndexError yükseltilecektir value/idx_value # idx_value == 0 ise, ZeroDivisionError yükseltilecektir 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 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("Ne olduğundan emin değilim, bu yüzden devam etmek güvenli değil, \ app will be interrupted") raise ex
Burada Exception'ın son except deyiminde kullanıldığını görüyoruz. Bunun nedeni Exception istisna nesnesinin herhangi bir istisna ile eşleşmesidir. Bu nedenle, Python bir eşleşme olduğunda diğer istisna işleyicilerini kontrol etmeyi bırakacağı için her zaman en sonda olmalıdır.
Yukarıdaki kodda, idx=5 , dolayısıyla arr[idx] yükseltecek IndexError Çünkü idx listenin uzunluğundan daha büyüktür arr
Ayrıca, uygulamanız tarafından hangi istisnanın yükseltildiğinden emin olmamak, yürütmeye devam etmek için asla güvenli değildir. Bu nedenle, tahmin edilmeyen istisnaları yakalamak için Exception türüne sahibiz. Ardından, kullanıcıyı bilgilendiririz ve aynı istisnayı yükselterek uygulamayı keseriz.
Try Else Deyimi
Bu bir isteğe bağlı özellik ve hata oluşmadığında çalışmasını istediğiniz kodu eklemenize olanak tanır. Bir hata oluşursa, bu else-block çalışmaz.
Aşağıdaki örnek Python kodunu düşünün, editörünüzü açın ve kodu elseTry.py olarak kaydedin
def fraction_of_one(divisor): value = 1/divisor # bölen sıfırsa, ZeroDivisionError yükseltilecektir return value if __name__ == '__main__': while True: try: # Kullanıcıdan girdi alın. # eğer girdi int() için geçerli bir argüman değilse, ValueError yükseltilecektir divisor = int(input("Enter a divisor: ")) # kesri hesaplamak için fonksiyonumuzu çağırın value = fraction_of_one(divisor) except (ValueError,ZeroDivisionError): print("Girdi sıfır olamaz ve int() için geçerli bir değişmez olmalıdır. Lütfen tekrar deneyin!") else: print("Değer: ", değer) break
Kullanıcıdan girdi alırız ve bunu 1'i bölmek için kullanırız. Burada iki olası istisnamız var, geçersiz bir kullanıcı girdisi ValueError ve bir sıfır(0) neden olacak ZeroDivisionError . except deyimimiz bu hataları ele alır.
Şimdi, değerini yazdırmak istiyoruz değer . else bloğumuz, yalnızca try bloğumuz hatasız bir şekilde yürütülürse yazdırıldığından emin olur. Bu önemlidir çünkü try bloğumuzda bir hata oluşursa değer tanımsız olacaktır. Bu nedenle, ona erişmek başka bir hataya neden olacaktır.
Yukarıdaki kodu Python elseTry.py ile çalıştırın
Yukarıdaki çıktı, ilk girdi için şunları yazdığımızı göstermektedir 0 Bölenimiz 0 aldığından, 1/bölen zeroDivisionError için geçersiz olan ikinci girdimiz k idi. int (), dolayısıyla istisna ValueError yükseltilir.
Ancak son girdimiz geçerli olan 9 idi ve sonuç olarak " değer " 0.11111111111111 olarak yazdırıldı
Nihayet İfadesini Deneyin
Bu aynı zamanda bir isteğe bağlı özellik ve istisna işleyicilerinde ne olursa olsun her zaman çalışacaktır.
İşte bu:
- Bir istisnanın gerçekleşip gerçekleşmediği
- Diğer bloklarda bir 'return' çağrısı yapılsa bile.
- Komut dosyası diğer bloklarda bırakılsa bile
Yani, her durumda çalışmasını istediğimiz bir kodumuz varsa, finally-block bizim adamımızdır. Bu blok çoğunlukla dosyaları kapatmak gibi temizlik işlemleri için kullanılır.
Aşağıdaki örnek Python kodunu ele alalım
def readFile(file_path): try: openFile = open(file_path,'r') # Dosyayı salt okunur olarak aç print(openFile.readline()) # Dosya içeriğinin ilk satırını oku except FileNotFoundError as ex: print(ex) finally: print("Temizleniyor...") openFile.close() if __name__ == '__main__': filePath = './text.txt' readFile(filePath)
Bu kod, geçerli dizindeki text.txt dosyasını açmaya ve okumaya çalışır. Dosya varsa, programımız dosyanın ilk satırını yazdıracak ve ardından finally bloğumuz çalışacak ve dosyayı kapatacaktır.
Diyelim ki bu program dosyasının bulunduğu dizinde text.txt adında bir dosyamız var ve içinde Hello var. Programı çalıştırırsak şu çıktıyı alırız
Bu örnek bilinçli olarak seçildi çünkü finally bloğundaki dosyaları kapatırken ortaya çıkabilecek küçük bir sorunu ele almamızı istedim.
Dosya mevcut değilse, istisna FileNotFoundError yükseltilecek ve değişken openFile tanımlanmayacaktır ve bir dosya nesnesi olmayacaktır. Bu nedenle, finally bloğu içinde kapatmaya çalışmak bir istisna oluşturacaktır UnboundLocalError 'in bir alt sınıfı olan İsimHatası .
Bu temel olarak şu değişkene başvurmaya çalıştığımızı söyler openFile atanmadan önce.
Buradaki küçük bir hile, finally bloğu içinde istisna işleyicileri kullanmaktır.
def readFile(file_path): try: openFile = open(file_path,'r') # Dosyayı salt okunur olarak aç print(openFile.readline()) # Dosya içeriğinin ilk satırını oku except FileNotFoundError as ex: print(ex) finally: try: print("Temizleniyor...") openFile.close() except: # tüm istisnaları yakalar pass # Bu hatayı görmezden gelin çünkü umurumuzda değil. if __name__ == '__main__': filePath = './text.txt' readFile(filePath)
Eğer try bloğumuz FileNotFoundError hatası verirse, aşağıdaki çıktıyı alırız
İstisnayı Yükselt
Python istisnaları ile ilgili iyi bir haber, onları kasıtlı olarak yükseltebilmemizdir. ifade yükseltme .
raise deyimi aşağıdaki sözdizimine sahiptir:
raise [ExceptionName[(*args: Object)]
Bir terminal açın ve Python dahili İstisnalar'dan herhangi bir istisna nesnesini yükseltin. Örneğin, eğer ZeroDivisionError yükseltirsek:
>>> raise ZeroDivisionError("Can't divide by zero")
Geri dönüşü alacağız:
Peki, istisnaları artırmak neden önemlidir?
- Özel istisnalar ile çalışırken.
- Akıl sağlığı kontrolleri sırasında.
Özel İstisna Sınıfları
Özel bir istisna, ihtiyacınıza özgü hataları işlemek için oluşturduğunuz bir istisnadır. İşin püf noktası, nesneden türeyen bir sınıf tanımlıyoruz İstisna sonra istisna sınıfımızı yükseltmek için raise deyimini kullanırız.
Kullanıcı girdisini kontrol etmek ve girdi değerinin negatif olmadığından emin olmak istediğimizi varsayalım (sanity check). Elbette, Python istisnası ValueError'ı yükseltebiliriz, ancak hatayı belirli ve kendini açıklayıcı bir ad vererek özelleştirmek isteriz InputIsNegativeError Ancak bu istisna Python'da yerleşik bir İstisna değildir.
Bu yüzden ilk olarak Exception'dan türetilecek temel sınıfımızı oluşturuyoruz.
class CustomError(Exception): "Bu modülün tüm istisnaları için temel sınıf istisnası" pass
Ardından, temel sınıfı miras alacak ve özel hatamızı ele alacak istisna sınıfımızı oluşturuyoruz.
class InputIsNegativeError(CustomError): """Kullanıcı negatif bir değer girdiğinde yükseltilir""" pass
Bunu test edelim
try: value = int(input()) if value <0: raise InputIsNegativeError # Değer negatifse istisna oluştur except InputIsNegativeError: # catch and handle exception print("Input value should't be negative")
Yukarıdaki kod kullanıcı girdisini talep eder ve negatif olup olmadığını kontrol eder. Doğruysa, daha sonra except-ifadesinde yakalanacak olan özel istisnamız InputIsNegativeError'ı yükseltir.
Kodun tamamı aşağıdadır:
class CustomError(Exception): "Bu modülün tüm istisnaları için temel sınıf istisnası" pass class InputIsNegativeError(CustomError): """Kullanıcı negatif bir değer girdiğinde bildirilir""" pass if __name__ == '__main__': try: value = int(input("Bir sayı girin: ")) if value <0: raise InputIsNegativeError # Değer negatifse istisna bildir except InputIsNegativeError: # catch and handle exceptionprint("Girdi değeri negatif olmamalı")
Eğer girdi değeri -1 gibi negatif bir sayı ise, o zaman çıktı alacağız:
Python özel istisnaları hakkında daha fazla ayrıntı için Python dokümanına göz atın.
Sıkça Sorulan Sorular
S #1) Python bir istisnayı nasıl ele alır?
Cevap ver: Python istisnaları şu şekilde ele alır try-except deyimi . İstisna oluşturabilecek kod bloğu deneyin ise blok hariç herhangi bir istisna ortaya çıkarsa bunu ele alacak kodu tutar.
S #2) Python'da bir istisnayı yükseltmek nedir?
Cevap ver: Python yorumlayıcısı geçersiz bir kodla karşılaştığında, Python'un bize beklenmedik bir şey olduğunu söylemenin kendi yolu olan bir istisna yükseltir. ifade yükseltme .
S #3) Python çoklu istisnaları nasıl ele alır?
Cevap ver: Python birden fazla istisnayı ya tek bir except bloğu ya da birden fazla except bloğu kullanarak ele alır.
Tek bir blok için, istisnalar bir tuple olarak aktarılır: hariç (Exception1, Exception2,...,ExceptionN) ve Python sağdan sola doğru bir eşleşme olup olmadığını kontrol eder. Bu durumda, her istisna için aynı işlem yapılır.
Tüm istisnaları yakalamanın bir başka yolu da except anahtar sözcüğünden sonra istisnanın adını atlamaktır.
except: # tüm istisnaları burada ele alın
İkinci yol ise her istisna için bir except bloğu kullanmaktır:
except Exception1: # Exception1'i işlemek için kod buraya gider except Exception2: # Exception2'yi işlemek için kod buraya gider except ExceptionN: # ExceptionN'i işlemek için kod buraya gider
Bu şekilde, her İstisna için ayrı eylemler gerçekleştirebilirsiniz.
S #4) Python'da İstisna işleme neden önemlidir?
Cevap ver: Python'da istisnaları ele almanın yararı, sağlam, temiz ve hatasız uygulamalar oluşturabilmemizdir. Üretim kodumuzun bazı hatalar nedeniyle çökmesini istemeyiz, bu nedenle hataları ele alır ve uygulamamızı çalışır durumda tutarız.
S #5) Python'da bir istisnayı nasıl yok sayarsınız?
Cevap ver: Python'da bir istisnayı yok saymak için geçmek except bloğundaki anahtar sözcük. Diyelim ki ValueError istisnasını yok saymak istiyoruz. Bunu şu şekilde yapacağız:
except ValueError: pass
Ne yaptığınızı bilmiyorsanız, istisnaları görmezden gelmek kötü bir uygulamadır. En azından, kullanıcıyı tüm olası hatalar hakkında bilgilendirin.
Sonuç
Bu eğitimde şunları ele aldık: Python İstisnaları, Traceback; ile istisnaların nasıl ele alınacağı Deneyin / Bunun dışında / Başka / Sonunda bloklar, nasıl yapılır Yükseltmek İstisnalar ve son olarak kendi Özel İstisnalarımızı nasıl oluşturacağımız.
Okuduğunuz için teşekkürler!