采摘处:http://blog.youkuaiyun.com/georgehuzz/article/details/5544697
问题: 你有一个长的处理过程, 该过程将诸塞GTK主循环对事件的控制, 导致界面在处理过程中没有响应, 更糟糕的是, 即使你想在处理过程中刷新界面, 提示进度都毫无用处, 因为它是诸塞的线程, 要么你看不见任何更新, 要么等到处理完成时他们一股脑儿都显示出来. 显然这不是我们想要的.
解决: google 了无数文章, 各种方法齐上阵, 但都显得麻烦, 最后, pygtk 的官方Q&A文档给出了最佳答案, 用生成器. 该文地址: http://faq.pygtk.org/index.py?req=show&file=faq23.020.htp
我的演示代码, 下面的代码演示一个在滚动窗口中按行输出0~99的数字. 用了空循环来模拟延时, 我们希望看到的是在界面中他们能一行一行的出现, 而不是"诸塞--dump"
import pygtk
pygtk.require('2.0')
import gobject
import gtk
import pango
gobject.threads_init()
class InfoWin:
def destroy(self, widget, data=None):
gtk.main_quit()
def __init__(self):
window = gtk.Window(gtk.WINDOW_TOPLEVEL)
window.connect("destroy", self.destroy)
window.set_border_width(10)
window.set_size_request(300,500)
box1 = gtk.VBox(False,0)
window.add(box1)
box1.show()
sw = gtk.ScrolledWindow()
sw.set_border_width(10)
sw.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
box1.pack_start(sw,True,True,0)
txtv = gtk.TextView()
txtv.modify_font(pango.FontDescription("sans 12"))
self.txtb = txtv.get_buffer()
sw.add(txtv)
sw.show()
txtv.show()
window.show()
def main(self):
gtk.main()
out=InfoWin()
def loopput():
for a in range(100):
for x in range(1000000):
pass
out.txtb.insert_at_cursor(str(a)+'/n')
yield True
yield False
lp=loopput()
gobject.idle_add(lp.next)
out.main()
关键点,
1. 把你的处理过程包裹成一个函数:function.
2. 用gobject.idle_add(function, next) 来告诉gtk我需要空闲处理这个函数.
3. 在你的处理过程中, 用yield True来告知主线程有空闲, 但是我这个子线程还没完. 当子线程跑完了, 用yield False告知主线程.
同样, 如果你是一个长的处理过程, 中间一般都要更新界面, 省得看上去和冻结了一样, 使用这种方法是最简单, 最安全, 没有副作用的. 参考官方文档的详细描述.