[英]Why does Tkinter image not show up if created in a function?
此代码有效:
import tkinter
root = tkinter.Tk()
canvas = tkinter.Canvas(root)
canvas.grid(row = 0, column = 0)
photo = tkinter.PhotoImage(file = './test.gif')
canvas.create_image(0, 0, image=photo)
root.mainloop()
它向我展示了图像。
现在,这段代码可以编译但没有显示图像,我不知道为什么,因为它是相同的代码,在 class 中:
import tkinter
class Test:
def __init__(self, master):
canvas = tkinter.Canvas(master)
canvas.grid(row = 0, column = 0)
photo = tkinter.PhotoImage(file = './test.gif')
canvas.create_image(0, 0, image=photo)
root = tkinter.Tk()
test = Test(root)
root.mainloop()
变量photo
是一个局部变量,它在类实例化后收集垃圾。 保存对照片的引用,例如:
self.photo = tkinter.PhotoImage(...)
如果您在“tkinter image doesn't display”上进行谷歌搜索,第一个结果是这样的:
为什么我的 Tkinter 图像没有出现? (常见问题解答目前尚未过时)
from tkinter import *
from PIL import ImageTk, Image
root = Tk()
def open_img():
global img
path = r"C:\.....\\"
img = ImageTk.PhotoImage(Image.open(path))
panel = Label(root, image=img)
panel.pack(side="bottom", fill="both")
but1 = Button(root, text="click to get the image", command=open_img)
but1.pack()
root.mainloop()
只需将 global 添加到 img 定义中即可
问题是 Python 通过称为垃圾收集的过程自动删除对变量的引用。 解决方案是保存参考或创建新参考。
以下是方法:
self
增加引用计数并保存引用。import tkinter
class Test:
def __init__(self, master):
canvas = tkinter.Canvas(master)
canvas.grid(row = 0, column = 0)
self.photo = tkinter.PhotoImage(file = './test.gif') # Changes here
canvas.create_image(0, 0, image=self.photo) # Changes here
root = tkinter.Tk()
test = Test(root)
root.mainloop()
import tkinter
l=[]
class Test:
def __init__(self, master):
canvas = tkinter.Canvas(master)
canvas.grid(row = 0, column = 0)
photo = tkinter.PhotoImage(file = './test.gif')
l.append(photo)
canvas.create_image(0, 0, image=photo)
root = tkinter.Tk()
test = Test(root)
root.mainloop()
使用方法 2 时,您可以像我一样制作一个全局列表,也可以在类中使用列表。 两者都会起作用。
一些有用的链接:
根据经验,无论何时在缩进代码块中创建图像,都需要保护对该图像的引用。 这是因为 python 的自动垃圾收集,当它销毁/离开该框架/页面/缩进代码块时,它会收集引用计数为 0 的所有内容。
处理它的规范方法是在全局命名空间的某处有一个图像列表,并将您的图像引用添加到该列表。 这很方便但效率不高,应该用于小型应用程序。
import tkinter as tk
global_image_list = []
global_image_list.append(tk.PhotoImage(file = 'test.png'))
正如 Bryan 在他的回答中所建议的那样,一种更有效的方法是将一个属性绑定到为您保存该引用的小部件或类。 如果你做self.image
或widget.image
之前分配的widget = tk.Widget(..
被销毁并收集垃圾。
import tkinter as tk
root = tk.Tk()
label = tk.Label(root, text='test')
label.image = tk.PhotoImage(file = 'test.png')
label.configure(image=label.image)
只需将global photo
添加为函数内的第一行。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.