繁体   English   中英

Python Tkinter按钮回调

[英]Python Tkinter button callback

当我单击由for循环创建的每个按钮时,我正在尝试打印出按钮编号。 以下是我的尝试。

import Tkinter as tk
root=tk.Tk()

def myfunction(a):
        print a

for i in range(10):
    tk.Button(root,text='button'+str(i),command=lambda:myfunction(i)).place(x=10,y=(10+(25*i)))
root.mainloop()

但不是打印出每个按钮编号,而是每次都给我最后一个按钮编号。 有什么我可以这样做,当我点击按钮1,它将打印1,2为2,依此类推?

简单的解决方法是每次创建lambda函数时使用当前值i初始化lambda函数。 这可以使用另一个虚拟变量j的Python默认值来完成。

command = lambda j=i: myfunction(j)

Blender的答案是一个聪明的解决方案,但如果你被函数抽象抛弃,这是另一种可行的方法。 它实际上只是创建一个保存在buttons的映射,从Button小部件到正确的数字。

import Tkinter as tk
root = tk.Tk()

def myfunction(event):
    print buttons[event.widget]

buttons = {}
for i in range(10):
    b = tk.Button(root, text='button' + str(i))
    buttons[b] = i # save button, index as key-value pair
    b.bind("<Button-1>", myfunction)
    b.place(x=10,y=(10+(25*i)))
root.mainloop()

这是因为i在你的匿名函数中引用的是计数器变量,而不是值:

from __future__ import print_function

x = [lambda: print(i) for i in range(10)]

for f in x:
    f()

这产生了连续9秒的输出。

要解决这个问题,你必须使用两个lambda和shadow i第二个函数(它创建你的第一个函数):

from __future__ import print_function

x = [(lambda i: (lambda: print(i)))(i) for i in range(10)]

for f in x:
    f()

虽然在那时,你最好只做一个命名函数:

def my_command(i):
    def inner_function():
        return my_function(i)

    return inner_function

并使用它像这样:

tk.Button(root, text='button' + str(i), command=my_command(i))

问题是tk中的Button命令忽略了任何参数,所以像mycommand(data)这样的东西会被忽略。

我使用了很多按钮,并决定将tk Button子类化为包含数据。 我称之为DataButton并添加了一个数据成员(在本例中是一个索引号)。 这样点击时会传回数据。 (索引号)现在我每次都希望它使用DataButton来保存索引甚至消息之类的信息。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM