【问题标题】:Why is the command bound to a Button or event executed when declared?为什么命令绑定到按钮或声明时执行的事件?
【发布时间】:2022-01-10 20:59:18
【问题描述】:

我的代码是:

from Tkinter import *

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

b = Button(admin, text='as', command=button('hey'))
b.pack()
mainloop()

按钮不起作用,它在没有我的命令的情况下打印“嘿”和“het”一次,然后,当我按下按钮时,什么也没有发生。

【问题讨论】:

  • @Mike-SMT 这就是原因。我想奖励那些对常见问题发表好的答案的人——尤其是如果问题很简单的话。许多人对简单的问题发表半生不熟的、没有动机的答案。我希望人们意识到,您不必成为编程专家也能写出出色的答案。

标签: python tkinter


【解决方案1】:

您需要创建一个可以用作命令的不带参数的函数:

b = Button(admin, text='as', command=lambda: button('hey'))

请参阅this document 的“将参数传递给回调”部分。

【讨论】:

    【解决方案2】:

    考虑这段代码:

    b = Button(admin, text='as', command=button('hey'))
    

    和这个完全一样:

    result = button('hey')
    b = button(admin, text='as', command=result)
    

    同样,如果您创建这样的绑定:

    listbox.bind("<<ListboxSelect>>", some_function())
    

    ...和这个是一样的:

    result = some_function()
    listbox.bind("<<ListboxSelect>>", result)
    

    command 选项采用对函数的引用,这是一种奇特的说法,您需要将函数的名称传递给它。要传递引用,您必须仅使用名称,而不使用括号或参数。例如:

    b = Button(... command = button)
    

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

    • 您可以创建一个中间函数,无需您的参数即可调用,然后调用您的 button 函数,
    • 您可以使用lambda 创建所谓的匿名函数。在任何方面它都是一个函数,只是它没有名字。当您调用 lambda 命令时,它会返回一个 reference 到创建的函数,这意味着它可以用于按钮的 command 选项的值。
    • 您可以使用functools.partial

    对我来说,lambda 是最简单的,因为它不需要像 functools.partial 那样的任何额外导入,尽管有些人认为 functools.partial 更容易理解。

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

    lambda: button('hey')
    

    你最终得到的函数在功能上等同于:

    def some_name():
        return button('hey')
    

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

    b = Button(... command = lambda: button('hey'))
    

    一般来说,这个网站上有很多有趣的 cmets 关于 lambda 的问题。请参阅问题Why Python lambdas are useful?。当您需要将变量传递给回调时,同样的讨论有an answer that shows how to use lambdas in a loop

    最后,请参阅标题为 Tkinter Callbackszone.effbot.org 文章以获得不错的教程。 lambda 的覆盖面非常少,但那里的信息可能仍然有用。

    【讨论】:

      【解决方案3】:

      图形用户界面示例:

      假设我有 GUI:

      import tkinter as tk
      
      root = tk.Tk()
      
      btn = tk.Button(root, text="Press")
      btn.pack()
      
      root.mainloop()
      

      按下按钮时会发生什么

      看到当btn被按下时,它会调用它自己的函数,这与下面例子中的button_press_handle非常相似:

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

      与:

      button_press_handle(btn['command'])
      

      你可以简单地认为command选项应该设置为,对我们要调用的方法的引用,类似于button_press_handle中的callback


      按下按钮时调用方法(Callback

      没有参数

      所以如果我想在按下按钮时print 某些东西,我需要设置:

      btn['command'] = print # default to print is new line
      

      密切注意()缺少print 方法省略的意思是:“这是我希望你调用的方法的名称按下 不要立即调用它。” 但是,我没有为 print 传递任何参数,所以它打印了在不带参数调用时打印的任何内容。

      个参数

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

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

      按下按钮时调用多个方法

      没有参数

      您也可以使用lambda 语句来实现这一点,但这被认为是不好的做法,因此我不会在此处包含它。好的做法是定义一个单独的方法multiple_methods,它调用所需的方法,然后将其设置为按钮按下的回调:

      def multiple_methods():
          print("Vicariously") # the first inner callback
          print("I") # another inner callback
      

      个参数

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

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

      然后设置:

      btn['command'] = lambda arg="live", kw="as the" : a_new_method(arg, opt1=kw)
      

      从回调中返回对象

      还要进一步注意callback 不能真正return,因为它只在button_press_handle 内部调用callback() 而不是return callback()。它确实 return 不是 该函数之外的任何地方。因此,您应该修改在当前范围内可访问的对象。


      带有global 对象修改的完整示例

      下面的例子将调用一个方法,每次按下按钮时都会改变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()
      

      Mirror

      【讨论】:

        【解决方案4】:

        引擎在“... command = ...”行赋值时评估函数的结果

        “命令”期望返回一个函数,这就是为什么使用 lambda 可以完成这项工作,因为它正在创建一个异常函数,该函数在评估期间返回给“命令”。 您也可以编写自己的函数,它也可以完成这项工作。

        这是一个有 lambda 和没有 lambda 的例子:

        #!/usr/bin/python
        # coding=utf-8
        
        from Tkinter import *
        # Creation de la fenêtre principale (main window)
        Mafenetre = Tk()
        res1 = StringVar()
        res2 = StringVar()
        
        def isValidInput(obj):
            if hasattr(obj, 'get') and callable(getattr(obj, 'get')):
                return TRUE
            return FALSE
        
        
        # stupid action 2 (return 12 on purpose to show potential mistake)
        def action1(*arguments):
            print "action1 running"
            for arg in arguments:
                if isValidInput(arg):
                    print "input value: ", arg.get()
                    res1.set(arg.get())
                else:
                    print "other value:", arg
            print "\n"
            return 12
        
        
        # stupid action 2
        def action2(*arguments):
            print "action2 running"
            a = arguments[0]
            b = arguments[1]
            if isValidInput(a) and isValidInput(b):
                c = a.get() + b.get()
                res2.set(c)
                print c
            print "\n"
        
        
        # a stupid workflow manager ordered by name
        def start_tasks(*arguments, **keywords):
            keys = sorted(keywords.keys())
            for kw in keys:
                print kw, "plugged "
                keywords[kw](*arguments)
        
        
        # valid callback wrapper with lambda
        def action1_callback(my_input):
            return lambda args=[my_input]: action1(*args)
        
        
        # valid callback wrapper without lambda
        def action1_callback_nolambda(*args, **kw):
            def anon():
                action1(*args)
            return anon
        
        
        # first input string
        input1 = StringVar()
        input1.set("delete me...")
        f1 = Entry(Mafenetre, textvariable=input1, bg='bisque', fg='maroon')
        f1.focus_set()
        f1.pack(fill="both", expand="yes", padx="5", pady=5)
        
        # failed callback because the action1 function is evaluated, it will return 12. 
        # in this case the button won't work at all, because the assignement expect a function 
        # in order to have the button command to execute something
        ba1 = Button(Mafenetre)
        ba1['text'] = "show input 1 (ko)"
        ba1['command'] = action1(input1)
        ba1.pack(fill="both", expand="yes", padx="5", pady=5)
        
        # working button using a wrapper
        ba3 = Button(Mafenetre)
        ba3['text'] = "show input 1 (ok)"
        # without a lambda it is also working if the assignment is a function
        #ba1['command'] = action1_callback_nolambda(input1)
        ba3['command'] = action1_callback(input1)
        ba3.pack(fill="both", expand="yes", padx="5", pady=5)
        
        # display result label
        Label1 = Label(Mafenetre, text="Action 1 result:")
        Label1.pack(fill="both", expand="yes", padx="5", pady=5)
        # display result value
        resl1 = Label(Mafenetre, textvariable=res1)
        resl1.pack(fill="both", expand="yes", padx="5", pady=5)
        
        
        # second input string
        input2 = StringVar()
        f2 = Entry(Mafenetre, textvariable=input2, bg='bisque', fg='maroon')
        f2.focus_set()
        f2.pack(fill="both", expand="yes", padx="5", pady=5)
        
        # third test without wrapper, but making sure that several arguments are well handled by a lambda function
        ba2 = Button(Mafenetre)
        ba2['text'] = "execute action 2"
        ba2['command'] = lambda args=[input1, input2], action=action2: start_tasks(*args, do=action)
        ba2.pack(fill="both", expand="yes", padx="5", pady=5)
        
        # display result label
        Label2 = Label(Mafenetre, text="Action 2 result:")
        Label2.pack(fill="both", expand="yes", padx="5", pady=5)
        # display result value
        resl2 = Label(Mafenetre, textvariable=res2)
        resl2.pack(fill="both", expand="yes", padx="5", pady=5)
        
        Mafenetre.mainloop()
        

        【讨论】:

          【解决方案5】:

          我认为解决这个问题的最好方法是使用 lambda 函数。

          from tkinter import *
          admin= Tk()
          def button(an):
              print(an)
              print("het")
          b = Button(admin, text="as", command=lambda: button("hey"))
          b.pack()
          mainloop()
          

          如果不想使用 command 关键字,可以改用 .bind() 方法:

          from tkinter import *
          admin= Tk()
          def button(an):
              print(an)
              print("het")
          b = Button(admin, text="as")
          b.pack()
          b.bind("<Button-1>", lambda bb: button("hey"))
          mainloop()
          

          使用拥有要调用的子函数(至少 1 个参数)的母函数(无参数)是愚蠢的。

          只是和大家分享一下,这是我的一个节目:

          import tkinter
          window = tkinter.Tk()
          
          def plus_them(field_1, field_2, field_3):
              field_3.delete(0, 'end')
              num1 = 0
              num2 = 0
              try:
                  num1 = int(field_1.get())
                  num2 = int(field_2.get())
              except:
                  print("Exception occurs")
              else:
                  print("Continue")
              result = num1 + num2
              field_3.insert(tkinter.END, str(result))
              return result
          def minus_them(field_1, field_2, field_3):
              field_3.delete(0, 'end')
              num1 = 0
              num2 = 0
              try:
                  num1 = int(field_1.get())
                  num2 = int(field_2.get())
              except:
                  print("Exception occurs")
              else:
                  print("Continue")
              result = num1 - num2
              field_3.insert(tkinter.END, str(result))
              return result
          
          #Input Panel:
          label_1 = tkinter.Label(window, text="First Number:")
          label_1.grid(row=0, column=0)
          label_2 = tkinter.Label(window, text="Second Number:")
          label_2.grid(row=1, column=0)
          entry_1 = tkinter.Entry(window)
          entry_1.grid(row=0, column=1)
          entry_2 = tkinter.Entry(window)
          entry_2.grid(row=1, column=1)
          
          #Button Panel:
          button_1 = tkinter.Button(window, text="Plus")
          button_1.grid(row=2, column=0)
          button_2 = tkinter.Button(window, text="Minus")
          button_2.grid(row=2, column=1)
          
          #Answer Panel:
          label_3 = tkinter.Label(window, text="The Answer:")
          label_3.grid(row=3, column=0)
          entry_3 = tkinter.Entry(window)
          entry_3.grid(row=3, column=1)
          
          #Event Handling:
          button_1.bind("<Button-1>", lambda p: plus_them(entry_1, entry_2, entry_3))
          button_2.bind("<Button-1>", lambda m: minus_them(entry_1, entry_2, entry_3))
          
          #Window Stuff:
          window.title("Plus and Minus Calculator")
          window.mainloop()
          

          就是这样。

          【讨论】:

            猜你喜欢
            相关资源
            最近更新 更多