python – 在Tkinter中动态创建菜单. (lambda表达式?)

前端之家收集整理的这篇文章主要介绍了python – 在Tkinter中动态创建菜单. (lambda表达式?)前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我有一个menubutton,点击它时会显示一个包含特定字符串序列的菜单.正是这个序列中的字符串,我们直到运行时才知道,所以必须在那一刻生成弹出的菜单.这就是我所拥有的:
class para_frame(Frame):
    def __init__(self,para=None,*args,**kwargs):
        # ...

        # menu button for adding tags that already exist in other para's
        self.add_tag_mb = Menubutton(self,text='Add tags...')

        # this menu needs to re-create itself every time it's clicked
        self.add_tag_menu = Menu(self.add_tag_mb,tearoff=0,postcommand = self.build_add_tag_menu)

        self.add_tag_mb['menu'] = self.add_tag_menu

    # ...

    def build_add_tag_menu(self):
        self.add_tag_menu.delete(0,END) # clear whatever was in the menu before

        all_tags = self.get_article().all_tags()
        # we don't want the menu to include tags that already in this para
        menu_tags = [tag for tag in all_tags if tag not in self.para.tags]

        if menu_tags:
            for tag in menu_tags:
                def new_command():
                    self.add_tag(tag)

                self.add_tag_menu.add_command(label = tag,command = new_command)
        else:
            self.add_tag_menu.add_command(label = "<No tags>")

重要的部分是“if menu_tags:”下的东西 – 假设menu_tags是列表[‘stack’,’over’,’flow’].那么我想要做的就是这样:

self.add_tag_menu.add_command(label = 'stack',command = add_tag_stack)
self.add_tag_menu.add_command(label = 'over',command = add_tag_over)
self.add_tag_menu.add_command(label = 'flow',command = add_tag_flow)

其中add_tag_stack()定义为:

def add_tag_stack():
    self.add_tag('stack')

等等.

问题是,变量’tag’取值’stack’然后取值’over’等等,并且在调用new_command之前不会对它进行求值,此时变量’tag’就是’流’.因此,无论用户点击什么,添加标记始终是菜单上的最后一个标记.

我最初使用的是lambda,我认为可能明确定义上面的函数可能会更好.无论哪种方式,问题都会发生.我已经尝试使用变量’tag’的副本(使用“current_tag = tag”或使用复制模块),但这并没有解决它.我不知道为什么.

我的思绪开始徘徊于“eval”之类的东西,但我希望有人能想到一种不涉及这种可怕事情的巧妙方式.

非常感谢!

(如果相关,Tkinter .__ version__返回’$Revision:67083 $’,我在Windows XP上使用Python 2.6.1.)

解决方法

首先,你的问题与Tkinter没有任何关系;最好是将它简化为一段简单的代码来证明您的问题,这样您就可以更轻松地进行实验.这是我试验过的简化版本.我用代替字母代替菜单,以便轻松编写一个小测试用例.
items = ["stack","over","flow"]
map = { }

for item in items:
    def new_command():
        print(item)

    map[item] = new_command

map["stack"]()
map["over"]()
map["flow"]()

现在,当我们执行此操作时,正如您所说,我们得到:

flow
flow
flow

这里的问题是Python的范围概念.特别是,for语句不会引入新的范围,也不会引入新的项目绑定;所以每次循环都会更新相同的项变量,并且所有new_command()函数都引用同一个项.

您需要做的是为每个项目引入一个新的范围,并使用新的绑定.最简单的方法是将其包装在一个新的函数定义中:

for item in items:
    def item_command(name):
        def new_command():
            print(name)
        return new_command

    map[item] = item_command(item)

现在,如果您将其替换为前面的程序,您将获得所需的结果:

stack
over
flow

猜你在找的Python相关文章