8

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

4

4 回答 4

18

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

主要的警告是工人不能与 Tkinter 小部件交互。他们必须将数据写入队列,而您的主 GUI 线程必须轮询该队列。

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

于 2012-05-11T18:28:01.463 回答
5

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

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

import subprocess 

subprocess.Popen("yourimage.png")
于 2013-01-23T14:50:51.977 回答
3

据我所知,Tkinter 不喜欢玩其他线程。看到这篇文章...我需要一点关于 Python、Tkinter 和线程的帮助

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

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

于 2012-05-11T18:22:38.190 回答
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")
于 2014-03-22T19:12:53.360 回答