当我打开 Tkinter 小部件 window 的新实例时,Tkinter 小部件丢失了信息

Tkinter widgets lost information when I open a new instance of their window

在我的真实软件中,我有一个主菜单(主 window)和其他菜单(顶层 windows),用户可以使用放置在主菜单中的一些小部件打开这些菜单。似乎工作正常,主要 window 可以打开其他顶层 windows,但我看到了一个非常大的问题。当我为同一个菜单打开多个 window 时,除最后一个菜单外,所有菜单都丢失了放置在它们的小部件中的信息(在我的例子中,是 Entry 和 ComboBox 小部件)。让我们从一个简单的例子开始:

from tkinter import *
from tkinter import ttk

class MainWindow:
    def __init__(self):
    
        # load a "SecondWindow" object:
        self.obj=SecondWindow(self)
        
        # main window's gui:
        self.parent=Tk()
        self.parent.geometry("300x280+360+200")
        self.parent.title("main window")
        self.parent.configure(background="#f0f0f0")
        
        self.OkButton=ttk.Button(self.parent, text="open the second window", width=26, command=lambda:self.obj.GUI())
        self.OkButton.place(x=20, y=20)

        self.parent.mainloop() 
    
class SecondWindow:
    def __init__(self, mw):
        self.mw=mw
    
    def GUI(self):
        self.window=Toplevel(self.mw.parent)
        self.window.geometry("300x180+360+200")
        self.window.title("second window")
        self.window.configure(background="#f0f0f0")
            
        self.MenuSV=StringVar()
        self.MenuSV.set("test test test")
        self.MenuComboBox=ttk.Combobox(self.window, state="readonly", values=("ciao", "hola", "hello", "Salut"), textvariable=self.MenuSV)
        self.MenuComboBox.place(x=20, y=20)
        
        self.window.mainloop()

# start the program:
if __name__ == "__main__":
    my_gui=MainWindow()

这段代码就像我真正的软件一样工作。在打开主window之前,加载了一个SecondWindow对象(他的主要组件是GUI功能)。当您打开第二个 window 一次(使用之前加载的 SeconWindow 对象)时,没问题,没有问题,但是如果您打开另一个,第一个丢失了放置在他的小部件中的信息。为什么?

我真的不明白这种奇怪的行为。我该如何解决这个问题?

由于你只在MainWindow中创建了一个SecondWindow()实例,所以每当SecondWindowGUI()被执行时,self.MenuSV将被重新分配另一个实例StringVar(),因此先前创建的 StringVar() 实例没有对其的变量引用,它已被垃圾回收。

您可以在单击按钮时创建 SecondWindow() 的新实例:

class MainWindow:
    def __init__(self):
    
        # load a "SecondWindow" object:
        #self.obj=SecondWindow(self)
        
        # main window's gui:
        self.parent=Tk()
        self.parent.geometry("300x280+360+200")
        self.parent.title("main window")
        self.parent.configure(background="#f0f0f0")
        
        self.OkButton=ttk.Button(self.parent, text="open the second window", width=26,
                                 command=lambda:SecondWindow(self).GUI()) # create new instance of SecondWindow here
        self.OkButton.place(x=20, y=20)

        self.parent.mainloop() 

或者在 SecondWindow 中保留对 self.MenuSV 的引用:

class SecondWindow:
    def __init__(self, mw):
        self.mw=mw
    
    def GUI(self):
        self.window=Toplevel(self.mw.parent)
        self.window.geometry("300x180+360+200")
        self.window.title("second window")
        self.window.configure(background="#f0f0f0")
            
        self.MenuSV=StringVar()
        self.MenuSV.set("test test test")
        self.MenuComboBox=ttk.Combobox(self.window, state="readonly", values=("ciao", "hola", "hello", "Salut"), textvariable=self.MenuSV)
        self.MenuComboBox.place(x=20, y=20)
        self.MenuComboBox.MenuSV = self.MenuSV  # keep a reference
        
        self.window.mainloop()