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

问题描述

在我的真实软件中,我有一个菜单(主窗口)和其他菜单(主窗口),用户可以使用放置在主菜单中的一些小部件打开它们。看起来工作,主窗口可以打开其他顶级窗口,但我看到一个非常大的问题。当我为同一个菜单打开多个窗口时,除了最后一个之外,所有窗口都丢失了放置在其小部件中的信息(在我的情况下,是 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()

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

enter image description here

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

解决方法

由于您只在 SecondWindow() 内创建了一个 MainWindow 实例,所以每当 GUI()SecondWindow 被执行时,self.MenuSV 将被重新分配 StringVar() 的另一个实例{1}},因此之前创建的 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() 

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

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()

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...