在Gtk python中线程化

时间:2012-08-12 14:37:12

标签: python multithreading gtk3

所以我忙着编写一个需要在一定时间后检查网站更新的应用程序,我使用python和Gtk +3

main.py文件

class Gui:
    ...
    def on_update_click():
        update()

app=Gui()
Gtk.main()

update.py文件

def update():
    #check site for updates
    time.sleep(21600) #check again in 6hrs

我怀疑我必须使用线程。 我的想法是:

Gtk.main()运行主线程。

当用户单击更新按钮时,update()在后台运行。 #thread 2

我的想法是正确还是错过了什么?

编辑: 好吧,
        on_update_click函数:

            Thread(target=update).start(). 

K,电脑不再冻结:D

所以现在发生的事情是,只有当我关闭Gtk.main()时,更新线程才会启动。当UI关闭时,它会继续更新,但我也希望在UI启动时启动它。

3 个答案:

答案 0 :(得分:6)

所以我终于设法让它发挥作用。 我需要说:

from gi.repository import Gtk,GObject

GObject.threads_init()
Class Gui:
    .....
    ......
    def on_update_click():
            Thread(target=update).start()

起初我用过:

thread.start_new_thread(update())
on_update_click函数中的

。如上所述我的J.F Sebastian这是不正确的,因为这会立即调用此线程。这冻结了我的整台电脑。

我刚刚补充说:

Thread(target=update).start()

on_update_clicked函数仅在主线程Gtk.main()关闭后才起作用。所以线程没有同时运行。

添加:     GObject.threads_init()

这允许线程以串行方式运行到python解释器: Threads in Gtk

答案 1 :(得分:4)

thread.start_new_thread(update())错了。它会立即在主线程中调用update(),您不应该直接使用thread模块;请改用threading模块。

您可以调用threading.current_thread()来找出执行update()的线程。

为了简化代码,您可以在主线程中运行所有gtk代码,并使用阻塞操作来检索网页并在后台线程中运行它们。

基于the extended example from GTK+ 3 tutorial

#!/usr/bin/python
import threading
import urllib2
from Queue import Queue

from gi.repository import Gtk, GObject

UPDATE_TIMEOUT = .1 # in seconds

_lock = threading.Lock()
def info(*args):
    with _lock:
        print("%s %s" % (threading.current_thread(), " ".join(map(str, args))))

class MyWindow(Gtk.Window):

    def __init__(self):
        Gtk.Window.__init__(self, title="Hello World")

        self.button = Gtk.Button(label="Click Here")
        self.button.connect("clicked", self.on_button_clicked)
        self.add(self.button)

        self.updater = Updater()
        self._update_id = None
        self.update()

    def on_button_clicked(self, widget):
        info('button_clicked')
        self.update()

    def update(self):
        if self._update_id is not None: 
            GObject.source_remove(self._update_id)

        self.updater.add_update(self.done_updating) # returns immediately
        # call in UPDATE_TIMEOUT seconds
        self._update_id = GObject.timeout_add(
            int(UPDATE_TIMEOUT*1000), self.update)

    def done_updating(self, task_id):
        info('done updating', task_id)
        self.button.set_label("done updating %s" % task_id)


class Updater:
    def __init__(self):
        self._task_id = 0
        self._queue = Queue(maxsize=100) #NOTE: GUI blocks if queue is full
        for _ in range(9):
            t = threading.Thread(target=self._work)
            t.daemon = True
            t.start()

    def _work(self):
        # executed in background thread
        opener = urllib2.build_opener()
        for task_id, done, args in iter(self._queue.get, None):
            info('received task', task_id)
            try: # do something blocking e.g., urlopen()
                data = opener.open('http://localhost:5001').read()
            except IOError:
                pass # ignore errors

            # signal task completion; run done() in the main thread
            GObject.idle_add(done, *((task_id,) + args))

    def add_update(self, callback, *args):
        # executed in the main thread
        self._task_id += 1
        info('sending task ', self._task_id)
        self._queue.put((self._task_id, callback, args))

GObject.threads_init() # init threads?

win = MyWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()

Gtk.main()

注意:GObject.idle_add()是唯一从不同线程调用的与gtk相关的函数。

另见Multi-threaded GTK applications – Part 1: Misconceptions

答案 2 :(得分:0)

线程是解决问题的第一种方法。您可以在该线程内创建线程并运行长时间运行的阻塞功能(并且您的GUI不会挂起)。

另一种方法是使用异步网络,例如使用python-gio(GObject-IO)或其他可以使用GLib主循环(like they do with Twisted)的库。这种方法有点不同,并使用非阻塞套接字操作。当来自套接字(您正在轮询的站点)的数据可供读取时,您的主循环将进行回调。遗憾的是,GIO没有高级HTTP API,因此您可以使用GSocketClient并手动创建HTTP请求结构。