Doç. Dr. Özgür Baştürk
Ankara Üniversitesi, Astronomi ve Uzay Bilimleri Bölümü
obasturk at ankara.edu.tr
http://ozgur.astrotux.org
Santigrad dereceyi Fahrenheit dereceye çeviren fonksiyonumuzu hatırlayalım. Bu fonksiyonu çağırırken istediğimiz değer(ler)i kaynak kodun içerisinden göndermek yerine kullanıcının girmesine izin verecek bir yapı işimizi oldukça kolaylaştırır ve kaynak kodumuzu farklı her değer için değiştirmemizi de önler. Bu tür bir yapıyı $input$ fonksiyonunu kullanarak kurabiliriz. $input$ fonksiyonu metin değişken döndürdüğünden bu değişkeni ihtiyaca göre diğer değişken tiplerine ($int$, $float$, $list$, $tuple$ …) çevirmek gerekebilir.
def F(C):
return (9.0/5)*C + 32
sant = input('C=? ')
sant = float(sant)
fahr = F(sant)
print(fahr)
$eval$ fonksiyonu bir metin değişkeni alıp, bir Python ifadesi şekinde yorumlar.
r = eval('1 + 2')
print(r, type(r))
r = eval('"Bu Python bir harika dostum!"')
print(type(r))
r = eval('[1, 6, 7.5]')
print(r, type(r))
from math import sqrt
r = eval('sqrt(2)')
print(r, type(r))
$eval$ fonksiyonu sadece metin değişken döndüren $input$ fonksiyonu ile kullanıcı tarafından girilen verinin herhangi bir tür değiştirme işlemine gerek kalmaksızın yorumlanmasını sağladığından oldukça pratiktir.
i1 = eval(input('Birinci sayi: '))
i2 = eval(input('Ikinci sayi: '))
r = i1 + i2
print('{:} + {:} toplami {:}\ndegeri ise {:}'.format(type(i1), type(i2), type(r), r))
$eval$ fonksiyonunun bir diğer kullanışlılığı kullanıcının formül girmesine ve girilen formülün bir Python ifadesi olarak yorumlanmasına imkan sağlamasıdır.
# butun matemtaik fonksiyonlarini indirelim ki kullanicinin
# girebilecegi her fonksiyona hazirlikli olalim.
from math import *
formul = input('x degiskenini iceren bir formul giriniz: ')
x = eval(input('x = '))
sonuc = eval(formul)
print('x={:g} icin {:s} = {:g}'.format(x, formul, sonuc))
Kullanıcı tarafından girilen bir formülü bir Python fonksiyonu olarak kullanmamızı sağlayan fonksiyondur.
from math import *
formul = input('x degiskenini iceren bir formul giriniz: ')
kod = """
def f(x):
return {:s}
""".format(formul)
print(kod)
exec(kod)
x = eval(input('x ='))
print(f(x))
Bu programın yaptığı iş kullanıcıdan aldığı ve sadece $x$ bağımsız değişkenini içeren herhangi bir ifadeyi bir Python fonksiyonuna cevirip döndürmektir. Söz konusu Python fonksiyonu $kod$ metin (string) değişkeni içerisinde oluşturulmaktadır. $exec$ fonksiyonu bu değişkenin içeriğini bir Python ifadesi olarak çalıştırmaktadır. İçerik bir fonksiyon tanımı olduğundan bu şekilde bir f(x) fonksiyonu tanımlanmış olur. Tanımlanan fonksiyon $sin(x).cos(3x) + x^{2}$ olduğunda örnekte verilen fonksiyon ile özdeştir.
Pek çok linux programı kullanıcıdan girdi (input) almak üzere komut satırını kullanır. Kullanıcı programın adının hemen arkasına programda kullanılacak girdi parametresinin değerini de girer. Girdi parametresi program tarafından bir salt metin değişkenine (string) atanır ve gerekli dönüşümler ve işlemler yapıldıktan sonra program çıktısını tekrar ekrana verir.
Bunun için $sys$ modülünde tüm komut satırı parametrelerini saklayan $argv$ listesi kullanılır. Bu listenin ilk elemanı ($argv[0]$) programın adıdır. İkinci eleman ($argv[1]$) ilk parametreyi, daha sonra gelen elemanlarsa sırayla diğer parametreleri saklar.
Bu bölümde yazılacak kodları ancak komut satırından (ya da jupyter'da %run "sihirli" sözcüğünü (magic word) kulllanarak) çaıştırabileceğiniz için kodların bir metin editörü ile .py uzantılı bir dosyaya yazılmış olması gerekir. Bu nedenle bu bölümde kullanılacak kodlar ders08_ornek_kodlar.tar.gz isimli bir sıkıştırılmış dosyada toplanmıştır. Bu dosyayı indirip, sıkıştırılmış dosyadaki klasörü (ders08_ornek_kodlar) kodları çalıştıracağınız yere (path) kopyalamanız durumunda aşağıdaki örnekler çalışacaktır.
Örneğin Fahrenheit dereceye dönüştürmek istediğimiz santigrad derece değerini kullanıcıdan komut satırı yoluyla girmesini istiyor olalım. Bunun için gerekli kod ders08_ornek_kodlar/ders08_orn01_C2F.py dosyasında bulunmaktadır. Kodu açıp inceleyiniz. Bu kodu Python 3.x çalıştıran herhangi bir terminalden
$ python ders08_orn01_C2F.py 25
komutu ile çalıştırabilecğeiniz gibi %run jupyter sihirli sözcüğüyle de çalıştırabilirsiniz. 25 burada Fahrenheit dereceye dönüştürmek istediğiniz sıcaklık değeri olup, istediğniz başka bir değeri de programa argüman olarak geçirebilirsiniz.
%run ders08_ornek_kodlar/ders08_orn01_C2F.py -25
Bir başka örnekte dikey atış probleminin parametrelerini kullanıcıdan komut satırı yoluyla almak üzere bir program yazmak istiyor olalım.
%run ders08_ornek_kodlar/ders08_orn02_dikeyatis.py 0.6 5.0
Bu durumda iki komut satırı argümanına ihtiyaç duyarız, sırasıyla bu argümanların değerleri $t$ ($argv[1]$) ve $v0$ ($argv[2]$) değişkenlerine salt metin olarak atanır. Bu değişkenlerin reel sayı olmasını istediğmiz için onları $float$ fonksiyonunu kullanarak öncelikle dönüştürmemiz gerekir. Daha sonra cismin düşey konumunu hesaplar ve ekrana yazdıırırız.
Bazen sayısı tam olarak bilinmeyen komut argümanıyla da çalışmak gerekbilir. Örneğin kullanıcının girdiği tüm (sayısı belirsiz) sayıları toplayan ve sonucu ekrana gerien bir kod yazmak istiyor olalım.
%run ders08_ornek_kodlar/ders08_orn03_toplama.py 3.2 -2.6 0.1 9.4 -4
Alternatif olarak aynı kodu isterseniz liste özelliklerini kullanarak çok daha kısa yazabilirsiniz.
%run ders08_ornek_kodlar/ders08_orn04_toplama2.py 3.2 -2.6 0.1 9.4 -4
Şu ana kadar gördüğünüz tüm komut satırından girdi parametreleri kullanıcının doğru sırada bu parametreleri girmesine bağlı olarak değer alırlar. Oysa ki (ls, cp, mv gibi başka Linux programlarında gördüğümüz gibi programın kullanıcının istenen parametreleri doğru sırada girmesinden bağımsız olarak -arguman = deger ikilileri ile çalışması çok iyi bir fikirdir.
Örnek olarak ivmeli bir hareketteki toplam yer değiştirmenin bulunması problemini ele alalım.
$$ s(t) = s_0 + v_0 t + \frac{1}{2} a t^2 $$İstenirse bu kod tüm parametreler için birer değer girilerek aşağıdaki şekilde çalıştırılabileceği gibi her bir parametrenin varsayılan değeri bilindiğinden sadece bazı parametrelerin değerleri girilip, diğer parametreler için varsayılan değerler kullanılarak da çalıştırılabilir.
%run ders08_ornek_kodlar/ders08_orn05_yerdegistirme.py --s0 5 --v0 3 --t 2 --a 2
Bu kodu şimdi sadece zaman parametresini (t) 5 saniye olarak komut satırından sağlamak suretiyle diğer parametreleri varsayılan değerlerinde (s0 = v0 = 0, a = 1) varsayarak çalştıralım ve cismin 5 saniye sonraki konumunu bu şekilde hesaplayalım.
%run ders08_ornek_kodlar/ders08_orn05_yerdegistirme.py --t 5
Hiçbir parametre için değer sağlamaz ve opsiyon ifadesini boş bırakırsak bu kez program tüm parametreler için varsayılan değerlerle (s0 = v0 = 0, t = a = 1) çalışacaktır.
%run ders08_ornek_kodlar/ders08_orn05_yerdegistirme.py
Diyelim ki kullanıcı programımızı çalıştırırken vermesi gereken bir komut satırı argümanını unuttu. Bu durumda $argv$ listesinin ulaşmaya çalıştığımız elemanı boş kalacağından IndexError: list index out of range şeklinde bir hata mesajı alır. Zira sadece programın adı girildiğinden $argv[0]$ doludur, ancak $argv[1]$ elemanı bulunmamaktadır.
Verilen hata mesajı bizim için açıklayıcı olsa da kullanıcı için olmayabilir. Kullanıcıya aşağıdaki şekilde bir mesajla hatasını söyleyip, yönlendirebiliriz. $sys.exit(1)$ programdan bir hata nedeniyle çıkıldığını belirtir. $sys.exit(0)$ programda bir hata olmadığı halde çıkmak için kullanılır. Bu fonksiyona verilen 0 haricindeki her şey 1 anlamına gelir.
Önce programımızı kullanıcının herhangi bir santigrad derece girdiği durum için çalıştıralım.
%run ders08_ornek_kodlar/ders08_orn06_C2F_v2.py 25
Şimdi kullanıcı sıcaklığı girmeyi unutmuş ya da programı doğru çalıştırmayı bilmediğinden sağlamamış olsun.
%run ders08_ornek_kodlar/ders08_orn06_C2F_v2.py
%run ders08_ornek_kodlar/ders08_orn06_C2F_v2.py 40
Hata yönetimine daha modern bir yaklaşım try – except yöntemini kullanmaktır. Bu yöntemi santigrad derece – fahrenheit derece dönüşüm kodumuza uyarlayabiliriz. Bu durumda da kodumuz aynı şekilde çalışacaktır.
%run ders08_ornek_kodlar/ders08_orn07_C2F_v3.py -6
%run ders08_ornek_kodlar/ders08_orn07_C2F_v3.py
%run ders08_ornek_kodlar/ders08_orn07_C2F_v3.py [1.2]
%run ders08_ornek_kodlar/ders08_orn07_C2F_v3.py eksialti
Bu yöntemde öncelikle argv listesinin birinci elemanının değeri C değişkenine atanmaya çalışılıyor (try bloku). Eğer kullanıcı komut satırı argümanı girmeyi unutmuşsa bu başarılamıyor ve kullanıcıya bir hata mesajı döndürülerek, sys.exit(1) komutu verildiği (except bloku) için programdan çıkılııyor ve bu nedenle programın gerisi çalışmıyor. Kullanıcı komut satırı argümanı girmiş ise C değişkeni değerini alıyor ve except bloğunu atlayarak çalışmaya Fahrenheit derece dönüşümünün hesabıyla devam ediyor ve ekrana çıktı verip, sonlanıyor.
Başka bir örnekle try - except yönteminin nasıl çalıştığını görelim.
def C2F(C):
return 9/5*C + 32
sant = '25'
try:
print("{:g} C = {:g} F".format(sant,C2F(sant)))
except:
print("Fahrenheit dereceye donusmesini istediginiz santigrad derece degerini nnumerik bir deger olarak girmediniz! Lutfen sicaklik icin santigrad derece cisinden numerik bir deger girerek tekrar deneyiniz!""")
sant2 = 25
print("-----------------------------")
try:
print("{:g} C = {:g} F".format(sant2,C2F(sant2)))
except:
print("Fahrenheit dereceye donusmesini istediginiz santigrad derece degerini nnumerik bir deger olarak girmediniz! Lutfen sicaklik icin santigrad derece cisinden numerik bir deger girerek tekrar deneyiniz!""")
Program $sant = '25'$ için çalıştırıldığında '25' bir string (metin) olduğu için hata verir ve bu nedenle $except$ bloğu çalışırken $sant2 = 25$ için çalıştırıldığında doğru bir şekilde çalışmaktadır.
IndexError Hatası: Python'da bir dizi değişkenin (liste, demet ya da metin (string)) uzunluğunun ötesinde bir elemana ulaşmaya çalışılınca Python'da bu hata “tetiklenir!”
veri = [1.0/i for i in range(1,10)]
print(len(veri))
veri[9]
Benzer şekilde sözlük değişkenlerde bulunmayan bir anathar "Key Error" hatası üretecektir.
notlar = {"15010018":91,"16010206":56, "1900254":34}
print(notlar)
notlar["12018921"]
ValueError Hatası: Örneğin bir metin değişkeni noktalı sayıya çevirmek gibi mümkün olmayan dönüşümlerin denenmesi Python'da bir değer hatası tetikler.
C = float('21 C')
NameError Hatası: Tanımlanmamış bir değişken ismine ulaşılmaya çalışıldığında tetiklenir.
print(degisken_adi)
ZeroDivisionError Hatası: 0 ile bölme bu hatayı tetikler.
3.0 / 0
SyntaxError Hatası: Yazımda yapılabilecek herhangi bir hata bu hata türünü tetikler.
print "Bu Python bir harika dostum!"
TypeError Hatası: Uygulanmak istenen İşlem ya da fonksiyona onunla uyumlu olmayan veri türü geçirildiğinde tetiklenir. Aşağıdaki örneği bir metin ve bir tam sayı için denerseniz hata vermez!
print("herhangi bir metin"*3.14)
Programda hata oluşturacak bir girdi durumunda kullanıcıya her farklı durum için açıklayıcı birer hata mesajı vermek iyi bir fikirdir. Aşağıdaki örneği inceleyelim. Sıcaklık değerini yanlışlıkla bir metin olarak verdiğimizde kodumuz bunu yorumlayıp bu duruma göre bir hata mesajı üretiyor.
%run ders08_ornek_kodlar/ders08_orn08_C2F_v4.py yirmibir
Kodumuza sıcaklık değeri olarak 21 dereceyi komut satırından sağladığımızda ise kodumuz doğru olarak çalışıyor.
%run ders08_ornek_kodlar/ders08_orn08_C2F_v4.py 21
Kodumuza mutlak sıfırın altında bir sıcaklık (örneğin -275 C) verildiğinde ise kodumuz kullanıcıya girdiği sıcaklık değerinin fiziksel bir sıcaklık değeri olmadığını hatırlatıyor.
%run ders08_ornek_kodlar/ders08_orn08_C2F_v4.py -275
Son olarak kullanıcı bir sıcaklık değeri girmezse buna uygun bir mesaj da kod tarafından üretiliyor.
%run ders08_ornek_kodlar/ders08_orn08_C2F_v4.py
Kullanıcıdan aldığnız bir sayısal değeri (tam ya da kayan noktalı sayı), $ y = 3 x^{4} - 2 x^{2} - x + 1$ polinomunda $x$ yerine koyarak, polinomdaki değerini hesaplayınız ve kullanıcıya sonucu gösteriniz. Kullanıcının $x$ için sayısal tek bir değer dışında (metin, liste, dizi gibi) bir türden değer girmesi durumunda kodunuzun nasıl çalıştığını anlatan bir mesajla uyarınız ve programdan çıkınız. Bu amaçla $try - except$ blokları yapısını kullanınız.
Kullanıcının komut satırından gireceği herhangi basit bir matematiksel ifadeyi ve yine kullanıcı tarafından girilecek bir değişken ($x$) için çalıştırarak sonucu ekrana getiren bir Python programı yazınız. Kodunuzun bir örnek çalışması aşağıdaki gibi olmaldıır.
Örnek Çalışma:
Formul giriniz: 2x - 1
x degerini giriniz: 5
x = 5 degeri icin 2x - 1 = 9.0
formülünü kullanarak cismin düşey konumunu hesaplayan ($y$) ekrana getiren bir Python kodu yazınız. Burada $g = 9.81 m/s^2$ 'dir. Ayrıca topun havada kaldığı süreyi ve çıktığı maksimum yüksekliği de hesaplayarak kullanıcıya gösteriniz. y'nin negatif değerler alamayacağı yerden (y = 0) atışları dikkate alarak; bu durumlarda kullanıcının girdiği değerlerin bir dikey atış için uygun olmadığını belirten bir hata mesajıyla programdan çıkılmasını sağlayınız.
datetime
modülünü import ederek $datetime.datetime.strptime()$ fonksiyonuyla $datetime$ nesnesine dönüştürünüz. Örnek:
tarih = 2021-01-01
tarih = datetime.datetime.strptime(tarih, "%Y-%m-%d")
Ardından Python'ın $datetime$ modülü $datetime$ nesnesi modülü metotlarından $weekday$ 'i kullanınız. Bu size tam sayı olarak (Pazartesi:0, Sali: 1, ..., Pazar:6) haftanın gününü verecektir. Bulduğunuz günü Türkçe ifadesiyle ekrana yazdırınız. Örnek bir çalışmayı aşağıda bulabilirsiniz.
Örnek Çalışma:
Tarihi giriniz: 2021-01-04
2021-01-04 bir Pazartesi gunudur.
Bir karacismin maksimum ışınım yaptığı dalgaboyu ($\lambda_{max}$) ile sıcaklığı ($T$) arasındaki ilişki aşağıdaki şekilde ifade edilebilir. Bu ifade Wien Kayma Yasası olarak bilinir.
$$ \lambda_{max} = \frac{0.002829 [m K]}{T [K]} $$Bir karacisimin maksimum ışınım yaptığı dalgaboyunu metre biriminde kullanıcıdan girmesini isteyen ve kullanıcıdan aldığı dalgaboyunu kullanarak Kelvin biriminde sıcaklığını ekrana yazdıran bir Python programı yazınız.
Birinci soruda yazdığınız programı biraz daha geliştirmek üzere, metre birmininde kendisine gönderilen bir karacismin maksimum ışınım yaptığı dalgaboyunu kullanarak Kelvin biriminde sıcaklığını hesaplayan ve çağrıldığı yere döndüren, $wien$ adında bir Python fonksiyonu yazınız. Fonksiyonunuz kayan noktalı ya da tamsayı, liste, demet değişken ve numpy dizileri olarak kendisine gönderilen dalgaboyu değer(ler)ine uygun olarak hesap yapmalı ve aynı türde geri döndürmelidir.
İkinci sorudaki çözümünüze uygun olarak kullanıcıdan dalgaboyunu yine metre (metre) biriminde ve nümerik olarak, ancak istediği türde (tam ya da kayan noktalı sayı, liste, demet ya da dizi) girmesini isteyen bir mesajla veri girişi isteyiniz. Aldığınız veriyi fonksiyonunuza gönderiniz ve fonksiyondan dönen sonuçları ekrana düzgün bir şekilde yazdırınız.
Üçüncü soru için yazdığınız kodunuzu geliştirmek üzere kullanıcı eğer
a) hiç bir veri girişi sağlamaz ya da istenen uygun türler dışında bir veri gönderdiğinde (sözlük ya da metin) kodunuzu nasıl kullanacağını anlatan basit bir mesajı ekrana gösterip çıkan;
b) liste ya da demet değişken dahilinde sayısal olmayan değerler de gönderdiğinde bu durumu kullanıcıya anlatan bir metni ekranına getirerek ne yapması gerektiğini anlatan basit bir mesajı ekrana gösterip çıkan;
c) doğru şekilde girilen bir türdeki verinin fonksiyonunuzdaki karşılığını hesapladıktan sonra ekrana düzgün bir şekilde ekrana getiren bir Python programı yazınız.
Programınız daha da geliştirmek üzere gönderilen dalgaboyunun elektromanyetik spektrumun hangi dalgaboyu bölgesine denk geldiğini hesaplayan bir fonksiyon daha yazınız. Kullanıcıya söz konusu karacismin ışınımının maksimumunu hangi dalgaboyu bölgesinde yaptığını da ekrana yazdırarak sağlaynız. Dalgaboyu bölgelerini belirlemek üzere ilgili wikipeida linkine başvurabilirsiniz.
Kodunuzla aynı yerde oluşturuacağınız veri_dosyalari klasörüne almanız gereken tayfsal_tur_sicaklik_kalibrasyonu.dat dosyasını okumak üzere açınız. Karacisim(ler)in sıcaklık değerine eşit yüzey sıcaklığındaki bir yıldızın dosyanın $T$ sütunundaki sıcaklık değerine uygun olarak, $TayfTuru$ sütununda verilen hangi tayf tür ya da türlerine (liste, demet ya da diziler için) karşılık geldiğini bulunuz ve ekrana yazdırınız.