Tkinter选项菜单小部件添加命令lambda不会产生预期的命令

时间:2017-02-14 09:42:11

标签: python python-2.7 lambda tkinter optionmenu

我正在尝试创建一个窗口,其中包含一个基于文件选择对话框填充的optionmenu。我能够使optionmenu包含基于先前选择的文件的所有选项。但是,无论用户选择了哪个选项,该变量都包含从所选文件中读取的最后一个选项。

如果有人能解释我为什么会这样,我会很高兴。因此,我在下面列出了一个最小的例子:

from Tkinter import *
Tk()

class App():

    chunks = ['A','B','C'] # These are read from a file in the actual program
    testVariable = StringVar()

    def __init__(self,master):

        def initSummary():
            self.dataButton["menu"].delete(0, 'end')
            for i in self.chunks:
                print i # To demonstrate what's in there
                self.dataButton["menu"].add_command(label=i, command = lambda: self.testVariable.set(i))

        top = self.top = Toplevel()
        top.protocol("WM_DELETE_WINDOW", lambda: close(self))
        self.sumButton = Button(top, text="Test", width=25, command=lambda: initSummary())
        self.sumButton.grid(row=0, column=0, sticky=W)
        self.dataButton = OptionMenu(top, self.testVariable, "Stuff")
        self.dataButton.grid(row=0, column=1, sticky=W)

# Call the main app
root = Tk()
app = App(root)
root.mainloop()

此代码将在选项菜单上显示C,无论用户选择了什么,而代码中的print会生成预期的AB和{ {1}}。我会理解C是否也显示了3次print,但是打印的内容与GUI中显示的内容不匹配会让我失望。

1 个答案:

答案 0 :(得分:-1)

我已经在for循环中解决了lambda的问题(感谢@ j_4321),添加了一个中间函数,如下面的代码所示。

from Tkinter import *

class App():

    chunks = ['A','B','C']

    def __init__(self,master):

        self.testVariable = StringVar()

        def refresh():
            # Prints what's in the variable
            print self.testVariable.get()

        def testFunc(x):
            return lambda: self.testVariable.set(x)

        def initSummary():
            self.dataButton["menu"].delete(0, 'end')
            for i in self.chunks:
                self.dataButton["menu"].add_command(label=i, command = testFunc(i))

        top = self.top = Toplevel()
        self.sumButton = Button(top, text="Test", width=25, command=lambda: initSummary())
        self.sumButton.grid(row=0, column=0, sticky=W)
        self.dataButton = OptionMenu(top, self.testVariable, "Stuff")
        self.dataButton.grid(row=0, column=1, sticky=W)
        self.testButton = Button(top, text="Variable", width=25, command = lambda: refresh())
        self.testButton.grid(row=1, column=0, sticky=W)

# Call the main app
root = Tk()
app = App(root)
root.mainloop()

如果这不是“正确的”,请发表评论。解决这个问题的方法。