使用 Multiprocessing 模块更新 Tkinter GUI

sar*_*jit 5 python tkinter multiprocessing

我一直在尝试使用 Multiprocessing 模块来更新 Tkinter GUI,但是当我运行此代码时,它给出了 Pickling 错误。

# Test Code for Tkinter with threads
import Tkinter
from multiprocessing import Queue
import multiprocessing
import time

# Data Generator which will generate Data
def GenerateData():
    global q
    for i in range(10):
        print "Generating Some Data, Iteration %s" %(i)
        time.sleep(2)
        q.put("Some Data from iteration %s \n" %(i))

def QueueHandler():
    global q, text_wid
    while True:
        if not q.empty():
            str = q.get()
            text_wid.insert("end", str)

# Main Tkinter Application
def GUI():
    global text_wid
    tk = Tkinter.Tk()
    text_wid = Tkinter.Text(tk)
    text_wid.pack()
    tk.mainloop()

if __name__ == '__main__':
# Queue which will be used for storing Data
    tk = Tkinter.Tk()
    text_wid = Tkinter.Text(tk)
    q = multiprocessing .Queue()
    t1 = multiprocessing.Process(target=GenerateData,args=(q,))
    t2 = multiprocessing.Process(target=QueueHandler,args=(q,text_wid))
    t1.start()
    t2.start()
    text_wid.pack()
    tk.mainloop()
Run Code Online (Sandbox Code Playgroud)

错误:

PicklingError: Can't pickle <type 'thread.lock'>: it's not found as thread.lock
Run Code Online (Sandbox Code Playgroud)

如果我删除了参数text_wid,则不会报告任何错误,但不会使用队列中的数据更新文本小部件。

更新 :

我修改了代码,以便在队列中有值时调用函数来更新 GUI,从而防止 Tkinter 小部件被传递到单独的进程。现在,我没有收到任何错误,但小部件没有更新数据。但是,如果我使用ThreadingMultiprocessing模块的混合,即创建一个单独的线程来处理队列中的数据,那么它就可以正常工作。我的问题为什么当我在单独的进程中运行处理程序代码时它不起作用。我是不是没有正确传递数据。下面是修改后的代码:

# Test Code for Tkinter with threads
import Tkinter
import multiprocessing
from multiprocessing import Queue
import time
import threading

# Data Generator which will generate Data
def GenerateData(q):
    for i in range(10):
        print "Generating Some Data, Iteration %s" %(i)
        time.sleep(2)
        q.put("Some Data from iteration %s \n" %(i))

def QueueHandler(q):
    while True:
        if not q.empty():
            str = q.get()
            update_gui(str)
            #text_wid.insert("end", str)

# Main Tkinter Application
def GUI():
    global text_wid
    tk = Tkinter.Tk()
    text_wid = Tkinter.Text(tk)
    text_wid.pack()
    tk.mainloop()

def update_gui(str):
    global text_wid
    text_wid.insert("end", str)

if __name__ == '__main__':
# Queue which will be used for storing Data
    tk = Tkinter.Tk()
    text_wid = Tkinter.Text(tk)
    q = multiprocessing.Queue()
    t1 = multiprocessing.Process(target=GenerateData,args=(q,))
    t2 = multiprocessing.Process(target=QueueHandler,args=(q,))
    t1.start()
    t2.start()
    text_wid.pack()
    tk.mainloop()
Run Code Online (Sandbox Code Playgroud)

cda*_*rke 4

你错过了一个重要的部分,你应该用陷阱保护你的通话__main__

if __name__ == '__main__': 
    q = Queue.Queue()
    # Create a thread and run GUI & QueueHadnler in it
    t1 = multiprocessing.Process(target=GenerateData,args=(q,))
    t2 = multiprocessing.Process(target=QueueHandler,args=(q,))

    ....
Run Code Online (Sandbox Code Playgroud)

请注意,队列作为参数传递,而不是使用全局参数。

编辑:刚刚发现另一个问题,您应该Queuemultiprocessing模块使用,而不是从Queue

from multiprocessing import Queue
Run Code Online (Sandbox Code Playgroud)