@ismailarilik’in bahsettiği gibi write
fonksiyonu olan bir veri tipi gerekiyor.
Normalde tk.Text()
widgetine yazı yazma fonksiyonu tk.Text().insert("insert", string)
. Dolayısıyla sys.stdout
'u, tk.Text()
widgetine doğrudan yönlendiremeyiz. Yönlendirebilmek için write()
metodu olan bir veri tipine ihtiyaç var. Bu veri tipini bir sınıf ile oluşturabiliriz. Bu sınıfın örnek metotlarının bir tanesi (write()
olan) tk.Text()
widgetinin tk.Text().insert("insert", string)
fonksiyonunu çalıştırır. Özetle bu sınıfın metotlarının isimleri tıpkı sys.stdout
'un metotlarının ismine benzer olursa, sys.stdout
'u sınıfa yönlendirebiliriz.
flush()
‘ı eklemezsek şöyle bir durumla karşılaşabiliriz:
Program çalışırken, sys.stdout.flush()
fonksiyonunu veya print(string, flush=True)
fonksiyonunu kullanırsak bir AttributeError
hatası alırız (AttributeError: ‘StdIOYonlendiricisi’ object has no attribute 'flush’).
Bir de, şu hatırlatmayı da yapmak isterim. Diyelim widgetin içine aşağıdaki kodları yazdık:
import time
count = 1
while count < 10:
print(count)
count += 1
time.sleep(1)
Bu kodların çıktısı, widgetin şu haliyle ancak 10 saniye sonra standart çıktılara yazdırılır ve bu arada program geçici olarak donar.
Bu durumu önlemenin bildiğim kadarıyla üç yolu var:
- Ya yukarıdaki kodları şöyle yazacağız:
import time
count = 1
while count < 10:
print(count, flush=True)
count += 1
time.sleep(1)
- Ya, aşağıdaki gibi olan
command1()
fonksiyonunu;
def command1():
exec(t1.get("1.0", "end"))
Şu şekilde değiştireceğiz:
def command1():
def inner_function():
exec(t1.get("1.0", "end"))
thread = threading.Thread(target=inner_function)
thread.daemon = True
thread.start()
exec()
fonksiyonunu bir iş parçacığının hedefi haline getirirsek, exec()
fonksiyonunun yürüttüğü işlem, Tkinter
uygulamasına engel olmaz.
- Ya da,
StdIOYonlendiricisi()
sınıfınınwrite()
metodunu şu şekilde değiştireceğiz:
def write(self, string):
self.text.insert("insert", string)
self.flush()
Yani her write()
metodu kullanımında aynı zamanda flush()
metodunu da kullanacağız.