在单独的线程中运行 Tkinter 表单
- 2025-03-26 09:09:00
- admin 原创
- 16
问题描述:
我编写了一个简短的模块,可以传递图像并简单地创建一个 Tkinter 窗口并显示它。我遇到的问题是,即使我在单独的线程中实例化并调用显示图像的方法,主程序也不会继续,直到 Tkinter 窗口关闭。
这是我的模块:
import Image, ImageTk
import Tkinter
class Viewer(Tkinter.Tk):
def __init__(self,parent):
Tkinter.Tk.__init__(self,parent)
self.parent = parent
self.initialize()
def initialize(self):
self.grid()
def show(self,img):
self.to_display = ImageTk.PhotoImage(img)
self.label_image = Tkinter.Label(self,image=self.to_display)
self.label_image.grid(column = 0, row = 0, sticky = "NSEW")
self.mainloop()
它似乎工作正常,除了当我从我的测试程序中调用它时(如下面的示例),它似乎不允许我的测试程序继续,即使在不同的线程中启动。
import Image
from viewer import Viewer
import threading
def showimage(im):
view = Viewer(None)
view.show(im)
if __name__ == "__main__":
im = Image.open("gaben.jpg")
t = threading.Thread(showimage(im))
t.start()
print "Program keeps going..."
我想也许我的问题是我应该在模块本身内创建一个新线程,但我只是想尝试保持简单,因为我对 Python 还不熟悉。
无论如何,提前感谢您的任何帮助。
编辑:为了清楚起见,我只是想制作一个可以在 Tkinter 窗口中显示图像的模块,这样我就可以随时使用此模块来显示图像。我遇到的问题是,任何时候程序使用此模块,它都无法恢复,直到 Tkinter 窗口关闭。
解决方案 1:
Tkinter 不是线程安全的,普遍的共识是 Tkinter 不能在非主线程中工作。如果您重写代码以使 Tkinter 在主线程中运行,则可以让您的工作程序在其他线程中运行。
主要警告是,工作者无法与 Tkinter 小部件交互。它们必须将数据写入队列,而主 GUI 线程必须轮询该队列。
如果您所做的只是显示图像,那么您可能根本不需要线程。线程仅在您有一个长时间运行的进程时才有用,否则该进程会阻塞 GUI。Tkinter 可以轻松处理数百个图像和窗口,而无需费力。
解决方案 2:
从你的评论来看,你似乎根本不需要 GUI。只需将图像写入磁盘并调用外部查看器即可。
在大多数系统上,应该可以使用以下命令启动默认查看器:
import subprocess
subprocess.Popen("yourimage.png")
解决方案 3:
据我所知,Tkinter 不喜欢在其他线程中播放。请参阅此帖子...我需要一些有关 Python、Tkinter 和线程的帮助
解决方法是在主线程中创建一个(可能隐藏的)顶层,生成一个单独的线程来打开图像等 - 并使用共享队列将消息发送回 Tk 线程。
您的项目需要使用 Tkinter 吗?我喜欢 Tkinter。它“快速而简单”。 - 但在许多情况下,其他 GUI 工具包才是最佳选择。
解决方案 4:
我曾尝试从单独的线程运行 tkinter,这不是一个好主意,它会冻结。有一个解决方案有效。在主线程中运行 gui,并将事件发送到主 gui。这是一个类似的例子,它只显示一个标签。
import Tkinter as t
global root;
root = t.Tk()
root.title("Control center")
root.mainloop()
def new_window(*args):
global root
print "new window"
window = t.Toplevel(root)
label = t.Label(window, text="my new window")
label.pack(side="top", fill="both", padx=10, pady=10)
window.mainloop()
root.bind("<<newwin>>",new_window)
#this can be run in another thread
root.event_generate("<<newwin>>",when="tail")
解决方案 5:
听起来你应该采用上面的解决方案,使用子进程作为编写程序的建议,可以基于 tkinter,作为单独的进程运行,显示存储在文件中的图像。这避免了线程并促进了无模式(非阻塞)显示。请注意,此解决方案避免了显示操作和主程序之间的线程或其他不便的交互。当我的程序的一部分使用 turtle(基于 tkinter 的图形)但不是屏幕阅读器友好的,而我的程序的另一部分使用 wxPython(基于 c++ 的图形框架)更适合屏幕阅读器时,我发现此解决方案很有用。目标用户是高度依赖屏幕阅读器友好应用程序的盲人。