while True:
= input("Bir sayı girin: ")
x if not x:
break
print(float(x)**2)
Python Programlamaya Giriş 22 - Hata yakalama, try/except
Hata yakalama (exception handling) beklenmedik durumlarda programınızın bir hata mesajı vermesi ve çalışmayı durdurması yerine, hataya kendi istediğimiz şekilde cevap vermesini sağlamanın bir yoludur. Hata yakalama Python programcılığının önemli bir parçasıdır, kaynak kodunu çok karışık hale getirmeden programınızın güvenilir bir şekilde çalışmasını sağlar.
Dizinin bütün yazılarına erişmek için Python Programlamaya Giriş kategorimize bakabilirsiniz. Bu dizideki yazılar ayrıca Jupyter defterleri halinde GitHub depomuzda da mevcut.
Hatalı girdiyi yakalamak
Bir örnekle başlayalım: Etkileşimli çalışarak kullanıcıdan sayılar alan ve aldığı sayıların karesini ekrana basan bir program yazalım. Boş satır okuduğunda program sonlansın.
Örnek olarak, programımız şöyle çalışabilir.
Bir sayı girin: 1
1.0
Bir sayı girin: -45.5
2070.25
Bir sayı girin: abc
---------------------------------------------------------------------------
ValueError Traceback (most recent call last)
<ipython-input-1-297c961843d7> in <module>()
3 if not x:
4 break
----> 5 print(float(x)**2)
ValueError: could not convert string to float: 'abc'
Son girdimiz "abc"
sayıya dönüştürülemediği için float()
fonksiyonu bir ValueError
hatası (Python terimiyle “exception”) verdi. Böyle hatalar programımızın çalışmasını durdurur. Oysa, bir hata yakalama (exception handling) yapısı kullanırsak bu tür sorunları programımızı durdurmadan halletmemiz mümkün olur. Söz gelişi:
while True:
= input("Bir sayı girin: ")
x if not x:
break
try:
= float(x)
y except ValueError:
print("Geçersiz sayı")
continue
print(y**2)
Bu program hatalı girdi verdiğimizde ekrana bir uyarı yazar ve tekrar girdi alır:
Bir sayı girin: 3
9.0
Bir sayı girin: -2
4.0
Bir sayı girin: abc
Geçersiz sayı
Bir sayı girin: 1.5
2.25
Bir sayı girin:
Bu programda, hata mesajı çıkarabilecek bölümü try:
blokunun içine aldık. Eğer float(x)
işlemi valueError
hatası verirse except ValueError
bloku çalıştırılır, ve kullanıcıya bir uyarı verilerek tekrar döngünün başına dönülür. Bu sayede program durmadan hatayı yakalayıp sorunu gidermiş oluruz.
Hata tipleri
Yukarıdaki örnekte ValueError
hatasını yakaladık, ama başka durumlardaki hata isimlerini nereden bileceğiz?
Öncelikle, yazdığınız her kod satırında neler olabileceğini düşünün. Hata durumu (exception) yaratan bir çok durum olabilir: Çağırdığınız fonksiyonda bir sayıyı sıfıra bölüyor olmanız mümkün mü? Bir matematiksel fonksiyona verdiğiniz değişken sayısal olmazsa ne olur? Üçüncü elemanını almaya çalıştığınız listede sadece iki eleman varsa? Açmak istediğiniz dosya diskte mevcut değilse?
Bu hata durumlarının ne olduğunu anlamak için komutları çalıştırıp ne tip hata aldığınıza bakabilirsiniz ve sonra buna göre try/except blokları yazabilirsiniz. Yardım belgelerinde de fonksiyonun hangi durumlarda hangi hataları yayınlayacağına dair bilgi mevcuttur.
Çalışma. Yukarıdaki hata durumlarını yaratan Python kodları yazın ve hangi hataların yayınlandığına bakın. Bu hataları bir try/except yapısı içine koyup uygun bulduğunuz şekilde düzenleyin.
Çalışma. open()
fonksiyonunun yardım belgelerine bakarak hangi durumda hangi hataların yayınlandığını inceleyin.
Python dilindeki ön tanımlı hataların tam listesini ve hangi durumlarda yayınlandıklarını resmi Python dökümanlarından okuyabilirsiniz.
Birden fazla hata durumu
Yukarıdaki örneğimizde, float()
fonksiyonuna yanlış parametre vermekle ortaya çıkan ValueError
hatasını yakalamıştık. Alternatif olarak şu kodu da kullanabilirdik:
while True:
= input("Bir sayı girin: ")
x if not x:
break
try:
= 1/float(x)
y except:
print("Geçersiz sayı")
continue
print(y)
Bir sayı girin: 0
Geçersiz sayı
Bir sayı girin: abc
Geçersiz sayı
Bir sayı girin: 2
0.5
Bir sayı girin:
Bu değişiklikle try
bloku içindeki herhangi bir hata ile except
bloku çalıştırılır. Ancak, bu yaklaşımda farklı hataların hepsi aynı except
blokuna yönlendirilir. Söz gelişi, yukarıda girdi olarak 0 verdiğimizde de ekrana "Geçersiz sayı"
yazılır. Oysa bu iki ayrı hata durumunun ayrı şekilde düzenlenmesini isteyebiliriz. O zaman iki farklı except
bloku kullanırız:
while True:
= input("Bir sayı girin: ")
x if not x:
break
try:
= 1/float(x)
y except ValueError:
print("Geçersiz sayı")
continue
except ZeroDivisionError:
print("Sıfıra bölme")
continue
print(y)
Bu program farklı hatalar için farklı uyarılar gösterir:
Bir sayı girin: abc
Geçersiz sayı
Bir sayı girin: 0
Sıfıra bölme
Bir sayı girin: 4
0.25
Bir sayı girin:
Hata durumu hiyerarşisi
Hata durumları bir nesne hiyerarşisi içinde tanımlanır. Bunların en geneli BaseException
sınıfıdır; diğer daha özelleşmiş hata durumları bunlardan türetilir.
Hata durumları hiyerarşisinin bir bölümü şöyledir (tam bir listeyi Python belgelerinde bulabilirsiniz):
BaseException
+-- SystemExit
+-- KeyboardInterrupt
+-- Exception
+-- StopIteration
+-- StopAsyncIteration
+-- ArithmeticError
| +-- FloatingPointError
| +-- OverflowError
| +-- ZeroDivisionError
+-- ImportError
| +-- ModuleNotFoundError
+-- LookupError
| +-- IndexError
| +-- KeyError
+-- OSError
+-- ValueError
Bu hiyerarşi sebebiyle, alt seviye bir hatayı yayınlayan bir kod, onun üstündeki hataları da yayınlar. Söz gelişi, 1/0
işlemi ZeroDivisionError
, ArithmeticError
, Exception
ve BaseException
hatalarının hepsine uyar.
Bir try/except yapısında bir hata durumu belirtmezsek en genel durum olan BaseException
yayınlanır.
try:
1/0
except:
print("Bir hata oldu.")
Bir hata oldu.
Ama böyle bir kullanım, okunaklı yazılım geliştirme açısından doğru değildir. Eğer try
blokumuz genişse ve birden fazla farklı hata olması ihtimali varsa, hepsi birden bu mesajı verir, ve hangi hatanın gerçekleştiğini tespit etmemiz mümkün olmaz.
try:
int("abc")
except:
print("Bir hata oldu.")
Bir hata oldu.
Belirsizliği azaltmak için en iyi yol, hiyerarşide en alt noktadaki (en dar kapsamlı) hata durumunu yakalamak ve ona göre ayrı except
blokları içinde sorunu gidermektir.
try:
2.5**1000
except OverflowError:
print("İşlem çok büyük.")
except ZeroDivisionError:
print("Sıfıra bölme.")
İşlem çok büyük.
Hatta, yaptığınız işlemin yeni bir hata durumu olmasını da sağlayabilirsiniz. Öntanımlı hata durumlarından yeni hatalar türetmeyi aşağıda göreceğiz.
Fonksiyonlarımızda hata durumu yayınlamak
Gördüğümüz gibi birçok Python fonksiyonu normal işleyişe uymayan durumlarda bir hata durumu yayınlıyor, ve programımızda bu hata durumunu yakalayarak işlem yapıyoruz. Kendi yazdığımız fonksiyonların içinde raise
komutu kullanarak bir hata durumu yayınlanmasını sağlayabiliriz. Örnek olarak, negatif argüman aldığında ValueError
yayınlayan bir faktöriyel fonksiyonu yazalım. Hata mesajını değiştirmemiz de mümkündür:
def faktöryel(x):
= int(x)
x if x<0:
raise ValueError("Negatif değer")
= 1
p for i in range(1,x+1):
*= i
p return p
Şimdi bu fonksiyonu bir try/except bloku içinde kullanalım.
for x in [5, -5, "abc", 5]:
try:
= faktöryel(x)
y except ValueError as e:
print(x,": ", e)
continue
print(y)
120
-5 : Negatif değer
abc : invalid literal for int() with base 10: 'abc'
120
Bu koddaki except ValueError as e:
komutu ile hata durumu e
isimli bir yerel değişkende saklanabilir ve blok içinde kullanılabilir. Yukarıdaki gibi print()
içinde kullanıldığında hata mesajını ekrana basarız. Negatif girdi ve harf girdisi durumlarında farklı hata mesajları çıktığına dikkat edin.
Yeni hata durumları yaratmak
Python’un standart hata durumlarına ek olarak, kendi hata durumlarımızı da yaratabiliriz. Yukarıda gördüğümüz hata durumu hiyerarşisi, aslında bir nesne hiyerarşisidir. Nesne sınıfları tanımlamayı sonraki bölümlerde göreceğiz, ama buradaki örneği nesne programlama bilmeden de uygulayabilirsiniz.
Yeni bir hata tanımlarken varolan bir hatayı temel alırız. Söz gelişi, genel Exception
nesne sınıfından türetilmiş bir VektörBoyuHatası
tanımlayalım.
class VektörBoyuHatası(Exception):
pass
Buradaki pass
kelimesi etkisiz bir komuttur. Python sözdizimi gereğince doldurulması gereken bir yere herhangi bir kod koymak istemediğimizde kullanırız.
Şimdi iki sayı listesinin iç çarpımını veren bir fonksiyon yazalım. Listeler aynı uzunlukta değilse iç çarpım tanımlı olmaz; bu durumda VektörBoyuHatası
yayınlayalım.
def iç_çarpım(L1, L2):
if len(L1)!=len(L2):
raise VektörBoyuHatası("Parametreler aynı sayıda elemandan oluşmalı.")
return sum( [a*b for (a,b) in zip(L1,L2)] )
1,2,3], [-1,0,1]) iç_çarpım([
2
1,2,3,4], [-1,0,1]) iç_çarpım([
VektörBoyuHatası: Parametreler aynı sayıda elemandan oluşmalı.
Bu fonksiyonu bir try/except yapısı içinde kullanabiliriz:
try:
1,2,3,4], [-1,0,1])
iç_çarpım([except VektörBoyuHatası as e:
print(e)
Parametreler aynı sayıda elemandan oluşmalı.