尝试使用
python创建计时器程序时遇到了问题.我希望它能让用户输入一定的时间让计时器倒计时并每0.1秒左右更新一次.到目前为止,我有这个代码:
from gi.repository import Gtk
import time
import threading
class TimerWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title = "Timer")
self.box = Gtk.Box(spacing = 1)
self.add(self.box)
self.entry = Gtk.Entry()
self.entry.connect("activate", self.start, self.entry)
self.box.pack_start(self.entry, True, True, 0)
def start(self, widget, entry):
starttime = time.time()
totaltime = float(entry.get_text())
self.update(starttime, totaltime, entry)
def update(self, starttime, totaltime, entry):
entry.set_text(str(totaltime - (time.time() - starttime)))
if float(entry.get_text()) > 0:
t = threading.Timer(0.1, self.update, [starttime, totaltime, entry])
t.start()
win = TimerWindow()
win.connect("delete-event", Gtk.main_quit)
win.set_keep_above(True)
win.show_all()
Gtk.main()
这似乎有点工作,但它有时会返回:
timer.py:31: Warning: g_object_ref: assertion 'object->ref_count > 0' failed
Gtk.main()
Segmentation fault
我不知道是什么原因引起的,我需要一些帮助.有人可以帮我找到阻止这种情况发生的方法吗?
最佳答案 你的程序崩溃是因为它从不同的线程调用GTK API,即
forbidden.幸运的是,很容易修改它才能正常工作 – 你可以使用线程,你只需要确保所有GTK调用都是从GUI线程,即运行主循环的线程.最简单的方法是让你的工作线程不直接执行GUI调用,而是使用GObject.idle_add将它们分派到主线程.因此,不是从计时器调用self.update,而是调用一个新方法schedule_update,它安排从GUI线程调用实际更新.由于未阻止GUI线程,因此更新将立即运行:
def update(self, starttime, totaltime, entry):
entry.set_text(str(totaltime - (time.time() - starttime)))
if float(entry.get_text()) > 0:
t = threading.Timer(0.1, self.schedule_update, [starttime, totaltime, entry])
t.start()
def schedule_update(self, *args):
GObject.idle_add(self.update, *args)
(您当然还需要从gi.repository导入GObject.)
不用说,使用GTK主循环超时(参见GObject.timeout_add)可以更好地实现这种调度,这不需要任何额外的调度,因为它们首先在GUI线程内执行回调.但是在合理的情况下使用线程是合适的 – 例如调用长时间运行的同步API,例如数据库访问而不冻结GUI,或执行内部释放GIL的计算.