在单独的线程中运行 Tkinter 表单

问题描述 投票:0回答:5

我编写了一个简短的模块,可以传递图像并简单地创建一个 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 窗口关闭。

python multithreading tkinter
5个回答
22
投票

Tkinter 不是线程安全的,普遍的共识是 Tkinter 不能在非主线程中工作。如果重写代码以便 Tkinter 在主线程中运行,则可以让工作线程在其他线程中运行。

主要警告是工作人员无法与 Tkinter 小部件交互。他们必须将数据写入队列,并且您的主 GUI 线程必须轮询该队列。

如果您所做的只是显示图像,则可能根本不需要线程。仅当您有一个长时间运行的进程(否则会阻塞 GUI)时,线程才有用。 Tkinter 可以轻松处理数百个图像和窗口,毫不费力。


5
投票

从您的评论看来,您根本不需要 GUI。只需将图像写入磁盘并调用外部查看器即可。

在大多数系统上,应该可以使用如下方式启动默认查看器:

import subprocess 

subprocess.Popen("yourimage.png")

3
投票

据我所知,Tkinter 不喜欢在其他线程中玩。 请参阅这篇文章...我需要一些有关 Python、Tkinter 和线程的帮助

解决方法是在主线程中创建一个(可能隐藏的)顶层,生成一个单独的线程来打开图像等 - 并使用共享队列将消息发送回 Tk 线程。

您的项目需要使用 Tkinter 吗? 我喜欢 Tkinter。 它“又快又脏”。 - 但在(很多)情况下,其他 GUI 工具包也是可行的方法。


2
投票

我尝试从单独的线程运行 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")

0
投票

听起来你应该采用上面的解决方案,使用子进程作为编写程序的建议,可以基于 tkinter,作为单独的进程运行,显示存储在文件中的图像。 这避免了线程化,允许无模式(非阻塞)显示。 请注意,这可以避免显示操作和主程序之间的线程或其他不方便的交互。 当我的程序的一部分使用turtle(基于tkinter的图形)但不适合屏幕阅读器,而另一部分使用wxPython(基于c++的图形框架)时,这个解决方案很有用,这对屏幕阅读器更友好。

© www.soinside.com 2019 - 2024. All rights reserved.