为什么在声明时执行Button参数"command"?

sal*_*alk 60 python tkinter

我的代码是:

from Tkinter import *

admin = Tk()
def button(an):
    print an
    print 'het'

b = Button(admin, text='as', command=button('hey'))
b.pack()
mainloop()
Run Code Online (Sandbox Code Playgroud)

按钮不起作用,在没有我的命令的情况下打印'hey'和'het'一次,然后,当我按下按钮时没有任何反应.

Bry*_*ley 75

command选项引用了一个函数,这是一种奇特的方式,表示你需要传递函数的名称.当你这样做时,button你正在调用该函数lambda,并且该结果将被赋予该lambda选项.

要传递引用,您必须仅使用名称,而不使用括号或参数.例如:

b = Button(admin, text='as', command=button('hey'))
Run Code Online (Sandbox Code Playgroud)

如果你想传递一个参数,如"嘿",你必须使用一些额外的代码:

  • 你可以创建一个中间函数,可以在没有你的参数的情况下调用,然后调用你的command函数,
  • 您可以使用lambda创建所谓的匿名函数.在任何方面它都是一个功能,除了它没有名称.当您调用该functools.partial命令时,它将返回对已创建函数的引用,这意味着它可用于functools.partial该按钮选项的值.
  • 你可以使用functools.partial

对我来说,这button是最简单的,因为它不需要任何额外的导入lambda,尽管有些人认为这command更容易理解.

要创建一个lambda使用参数调用函数的lambda函数,您可以执行以下操作:

result = button('hey')
b = button(admin, text='as', command=result)
Run Code Online (Sandbox Code Playgroud)

您最终得到的功能在功能上等同于:

b = Button(... command = button)
Run Code Online (Sandbox Code Playgroud)

正如我之前所说,command返回对这个无名函数的引用.由于引用是该button选项所期望的,因此您可以lambda直接在创建按钮时使用:

lambda: button('hey')
Run Code Online (Sandbox Code Playgroud)

这个网站上有一个问题,一般来说有很多关于lambda的有趣评论.请参阅问题为什么Python lambdas有用?.同样的讨论有一个答案,显示当你需要将变量传递给回调时如何在循环中使用lambdas.

最后,请参阅effbot.org上标题为Tkinter Callbacks的部分,以获得一个很好的教程.lambda的覆盖范围非常精简,但那里的信息可能仍然有用.


Luk*_*ský 11

您需要创建一个没有参数的函数,您可以将其用作命令:

b = Button(admin, text='as', command=lambda: button('hey'))
Run Code Online (Sandbox Code Playgroud)

请参阅本文档的"将参数传递给回调"部分.


Nae*_*Nae 5

GUI示例:

假设我有GUI:

import tkinter as tk

root = tk.Tk()

btn = tk.Button(root, text="Press")
btn.pack()

root.mainloop()
Run Code Online (Sandbox Code Playgroud)

按下按钮时会发生什么

看到btn按下时它会调用自己的函数,函数与button_press_handle以下示例非常相似:

def button_press_handle(callback=None):
    if callback:
        callback() # Where exactly the method assigned to btn['command'] is being callled
Run Code Online (Sandbox Code Playgroud)

与:

button_press_handle(btn['command'])
Run Code Online (Sandbox Code Playgroud)

您可以简单地认为command应该将option设置为对我们要调用的方法的引用,类似于callbackin中button_press_handle


按下按钮时调用方法(回调

没有参数

因此,如果要在print按下按钮时进行某些操作,则需要进行以下设置:

btn['command'] = print # default to print is new line
Run Code Online (Sandbox Code Playgroud)

请密切注意缺少()print方法的不足,该方法的含义是:“这是我要在按下时调用的方法名称,不要立即调用。” 但是,我没有为传递任何参数,print因此它在无参数调用时将打印出任何内容。

论点

现在,如果我还想将参数传递给要在按下按钮时调用的方法,我可以利用匿名函数,该函数可以使用lambda语句创建,在这种情况下,将使用print内置方法,如下所示:

btn['command'] = lambda arg1="Hello", arg2=" ", arg3="World!" : print(arg1 + arg2 + arg3)
Run Code Online (Sandbox Code Playgroud)

按下按钮时调用多种方法

没有参数

您也可以使用using lambda语句来实现该功能,但是这被认为是不好的做法,因此在此不再赘述。好的做法是定义一个单独的方法,multiple_methods该方法调用所需的方法,然后将其设置为按下按钮的回调:

def multiple_methods():
    print("Vicariously") # the first inner callback
    print("I") # another inner callback
Run Code Online (Sandbox Code Playgroud)

论点

为了将参数传递给调用其他方法的方法,请再次使用lambda语句,但首先:

def multiple_methods(*args, **kwargs):
    print(args[0]) # the first inner callback
    print(kwargs['opt1']) # another inner callback
Run Code Online (Sandbox Code Playgroud)

然后设置:

btn['command'] = lambda arg="live", kw="as the" : a_new_method(arg, opt1=kw)
Run Code Online (Sandbox Code Playgroud)

从回调返回对象

还要进一步注意,这callback并不是真的,return因为它仅在内部button_press_handlecallback()而不是一起调用return callback()。它没有,return不在该功能之外的任何地方。因此,您应该修改当前作用域中可访问的对象。


具有全局对象修改的完整示例

下面的示例将调用一个方法,该方法btn每次按下按钮都会更改的文本:

import tkinter as tk

i = 0
def text_mod():
    global i, btn           # btn can be omitted but not sure if should be
    txt = ("Vicariously", "I", "live", "as", "the", "whole", "world", "dies")
    btn['text'] = txt[i]    # the global object that is modified
    i = (i + 1) % len(txt)  # another global object that gets modified

root = tk.Tk()

btn = tk.Button(root, text="My Button")
btn['command'] = text_mod

btn.pack(fill='both', expand=True)

root.mainloop()
Run Code Online (Sandbox Code Playgroud)

镜子