python - 有哪些技术可以允许 tkinter 程序中的多个线程?
问题描述
我发现了这个简单的 Hello World tkinter 程序:
import tkinter as tk
class Application(tk.Frame):
def __init__(self, master=None):
super().__init__(master)
self.master = master
self.pack()
self.create_widgets()
def create_widgets(self):
self.hi_there = tk.Button(self)
self.hi_there["text"] = "Hello World\n(click me)"
self.hi_there["command"] = self.say_hi
self.hi_there.pack(side="top")
self.quit = tk.Button(self, text="QUIT", fg="red",
command=self.master.destroy)
self.quit.pack(side="bottom")
def say_hi(self):
print("hi there, everyone!")
self.hi_there["text"] = "Hello World\n(click me again)"
root = tk.Tk()
app = Application(master=root)
app.mainloop()
如果我想让该say_hi()
方法执行长时间运行的任务并偶尔更新 GUI,该怎么办?
如果我试试这个:
def say_hi(self):
print("hi there, everyone!")
self.hi_there["text"] = "Hello World\n(wait...)"
sleep(2) # pretend to do something long-running
self.hi_there["text"] = "Hello World\n(click me again)"
然后 GUI 在睡眠期间锁定,我从来没有看到按钮变为:Hello World\n(wait...)
解决方案
一种可能的解决方案是下面的修改Application
类,它使用queue
一个新方法poll()
作为消息泵来操作Thread
希望运行的代码项。
from time import sleep
from queue import Queue
from threading import Thread
class Application(tk.Frame):
def __init__(self, master=None):
super().__init__(master)
self.master = master
self.pack()
self.create_widgets()
self.queue = Queue()
self.poll()
def create_widgets(self):
self.hi_there = tk.Button(self)
self.hi_there["text"] = "Hello World\n(click me)"
self.hi_there["command"] = self.start_hi
self.hi_there.pack(side="top")
self.quit = tk.Button(self, text="QUIT", fg="red",
command=self.master.destroy)
self.quit.pack(side="bottom")
def say_hi(self):
def do_wait():
self.hi_there["text"] = "Hello World\n(wait...)"
def do_again():
self.hi_there["text"] = "Hello World\n(click me again)"
print("hi there, everyone!")
self.queue.put(do_wait)
sleep(2) # pretend to do something long-running
self.queue.put(do_again)
def start_hi(self):
thread = Thread(target = self.say_hi)
thread.start()
def poll(self):
if not self.queue.empty():
item = self.queue.get()
item()
self.master.after(100, self.poll)
如果您使用该版本的类,那么您会注意到按钮上的版本和文本按预期hi_there Button
更改。wait...
注意事项:此代码还允许您创建大量Thread
实例,但该poll()
方法仅每 100 毫秒轮询一次
推荐阅读
- javascript - 旋转图标引导程序 3
- nested-lists - 如何按列表中的第二个元素对嵌套列表进行排序,然后输出整个列表中第二低的元素?
- c# - Unity C# 调用在以下方法或属性之间不明确:`System.Math.Round(double, int)' 和 `System.Math.Round(decimal, int)
- html - 如何居中反应引导模式?
- vega - 使用分层条形图选择相同 x 位置的两个条形图
- javascript - 防止 JavaScript 中的值冲突
- r - 使用 ggplotly() 时不显示根据颜色、形状和填充创建的散点图
- docker - 从 Azure Pipeline Task 构建 docker 映像
- list - 在 Haskell 中分离矩阵的行和列
- node.js - 使用 lerna 作为 API 版本管理器