5

我有这个主线程:

桂皮

from gi.repository import Gtk, Gdk
import Process
import gobject

class gui():
    def __init__(self):
        self.window = Gtk.Window()
        self.window.connect('delete-event', Gtk.main_quit)

        self.box = Gtk.Box()
        self.window.add(self.box)

        self.label = Gtk.Label('idle')
        self.box.pack_start(self.label, True, True, 0)

        self.progressbar = Gtk.ProgressBar()
        self.box.pack_start(self.progressbar, True, True, 0)

        self.button = Gtk.Button(label='Start')
        self.button.connect('clicked', self.on_button_clicked)
        self.box.pack_start(self.button, True, True, 0)

        self.window.show_all()
        gobject.threads_init()

        Gdk.threads_enter()
        Gtk.main()
        Gdk.threads_leave()

    def working1():
        self.label.set_text('working1')
        t = Process.Heavy()
        t.heavyworks1() 
        self.label.set_text('idle') 

    def on_button_clicked(self, widget):
        Gdk.threads_enter()
        working1()
        Gdk.threads_leave()

if __name__ == '__main__':
    gui = gui()

此代码将生成此 gui: 图像

我有第二个模块来做逻辑。

进程.py

import threading

class Heavy(threading.Thread):
    def __init__(self):
        threading.Thread.__init__(self)

    def heavyworks1(self):
        #doing heavy works1
        #return result

   def heavyworks2(self, *param):
        #doing heavy works2
        #return result

当我执行此操作时,操作有效,但 gui 冻结了。如何做好?

编辑:

正如 user4815162342 所说,我将代码更改为:

from gi.repository import Gtk, Gdk, GLib
import Process
import gobject
import threading

class gui():
def __init__(self):
    self.window = Gtk.Window()
    self.window.connect('delete-event', Gtk.main_quit)

    self.box = Gtk.Box()
    self.window.add(self.box)

    self.label = Gtk.Label('idle')
    self.box.pack_start(self.label, True, True, 0)

    self.progressbar = Gtk.ProgressBar()
    self.box.pack_start(self.progressbar, True, True, 0)

    self.button = Gtk.Button(label='Start')
    self.button.connect('clicked', self.on_button_clicked)
    self.box.pack_start(self.button, True, True, 0)

    self.window.show_all()

    gobject.threads_init()
    GLib.threads_init()
    Gdk.threads_init()
    Gdk.threads_enter()
    Gtk.main()
    Gdk.threads_leave()

def init_progress(self, func, arg):
    self.label.set_text('working1')
    self.worker = threading.Thread(target=func, args=[arg])
    self.running = True
    gobject.timeout_add(200, self.update_progress)
    self.worker.start()

def update_progress(self):
    if self.running:
        self.progressbar.pulse()
    return self.running

def working(self, num):
    Process.heavyworks2(num)    
    gobject.idle_add(self.stop_progress)

def stop_progress(self):
    self.running = False
    self.worker.join()
    self.progressbar.set_fraction(0)
    self.label.set_text('idle') 

def on_button_clicked(self, widget):
    self.init_progress(self.working, 100000)

if __name__ == '__main__':
    gui = gui()

使用该代码,程序有时可以工作,但有时会出现此错误。

1.

**
Gtk:ERROR:/build/buildd/gtk+3.0-3.4.2/./gtk/gtktextview.c:3726:gtk_text_view_validate_onscreen: assertion failed: (priv->onscreen_validated)
Aborted (core dumped)

2.

*** glibc detected *** python: free(): invalid next size (fast): 0x09c9f820 ***

3.

Segmentation fault (core dumped)
4

2 回答 2

7

您实际上并没有启动线程,您只是实例化了一个可用于启动它的对象。完整的解决方案需要仔细分离 GUI 线程和工作线程之间的职责。您想要做的是以下内容:

  1. 在由 GUI 代码生成和连接的单独线程中进行繁重的计算。计算不应产生自己的线程,也不需要知道线程(当然,线程安全除外)。

  2. 线程完成后,用于gobject.idle_add()告诉 GUI 可以撤回进度指示器。(gobject.idle_add大约是唯一一个可以安全地从另一个线程调用的 GTK 函数。)

通过这样的设置,无论计算做什么,GUI 都会保持完全响应并更新进度条,并且保证 GUI 线程在计算完成时会注意到。关于您当前代码的另外两点:

  • 实例化threading.Thread而不是继承它。这样您就不必费心实施run(). 但是,在这两种情况下,您都必须调用thread.start()来启动线程。

  • 不要调用threads_enter()and threads_leave(),除非你真的知道你在做什么。请记住,只要您从单个线程(初始化 GTK 的同一线程)调用所有 GTK 函数,就可以了。

这是实现上述建议的概念验证代码:

    def working1(self):
        self.label.set_text('working1')
        self.work_thread = threading.Thread(self.run_thread)
        self.running = True
        gobject.timeout_add(200, self.update_progress)
        self.work_thread.start()
        # the GUI thread now returns to the mainloop

    # this will get periodically called in the GUI thread
    def update_progress(self):
        if self.running:
            self.progressbar.pulse()   # or set_fraction, etc.
        return self.running

    # this will get run in a separate thread
    def run_thread(self):
        Process.heavyworks1()      # or however you're starting your calculation
        gobject.idle_add(self.stop_progress)

    # this will get run in the GUI thread when the worker thread is done
    def stop_progress(self):
        self.running = False
        self.work_thread.join()
        self.label.set_text('idle')
于 2013-06-05T20:18:38.977 回答
0

正如您所建议的,您需要为此启动另一个线程。通常,python 中的线程非常简单,但使用 GUI 可能会变得棘手。

这应该有帮助:Python。使用 Gtk GUI 在后台做一些工作

于 2013-06-05T08:26:52.240 回答