Tkinter选项菜单将命令添加到多个选项菜单

2022-02-26 00:00:00 python tkinter lambda closures optionmenu

问题描述

基本上我有一系列在循环中创建的OptionMenus,但当前为空:

option_menu = []
for ii in range(jj):
    option_menu.append([])  
    for ll in range(kk):   
        option_menu[ii].append(OptionMenu(frame,tkinter_text_var[ii][ll],''))

然后在其他地方,我使用复选框来设置如下行的值:

for ii in range(jj):
    for ll in range(kk):   
        option_menu[ii][ll]["menu"].add_command(label = name_from_box.get(), command = lambda: tkinter_text_var[ii][ll].set(name_from_box.get()))

这可以正确填充所有OptionMenus,但是当我在任何OptionMenus中选择一个值时,它只设置OPTION_Menu[jj][kk](即最后设置的值)。

那么我做错了什么?


解决方案

这是一个涉及闭包的非常常见问题。请看以下示例:

alist = [lambda : x for x in range(10) ]
print (alist[2]()) #9
print (alist[4]()) #9

都是9。为什么?因为每个lambda函数都引用变量xx在循环的每次迭代中都会更改,但它们仍然引用同一个对象。

解决此问题的一种方法是使用默认参数。默认参数在创建函数时求值,而不是在调用时求值。

alist = [lambda y=x: y for x in range(10) ]
print (alist[2]()) #2
print (alist[4]()) #4

(做同样事情的另一种方式涉及functools.partial,您有时会看到.)

我经常说--"注意闭包"。它们可能有点棘手。

相关文章