采摘处:http://blog.csdn.net/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告知主线程.
同样, 如果你是一个长的处理过程, 中间一般都要更新界面, 省得看上去和冻结了一样, 使用这种方法是最简单, 最安全, 没有副作用的. 参考官方文档的详细描述.