Tkinter Python:如何通过OptionMenu中的lambda函数传递多个参数

2024-06-16 11:53:23 发布

您现在位置:Python中文网/ 问答频道 /正文

所以正如标题所示,当选项菜单更改时,我很难将2个参数传递给我想调用的函数。在

代码如下:

OPTIONS = [
        "Fire",
        "Ice",
        "Storm",
        "Life",
        "Myth",
        "Death",
        "Balance"
    ]

    var = StringVar(frame)
    var.set("Select School") # initial value

    option = OptionMenu(frame, var,*OPTIONS,command= lambda frame,var : changeSchool(frame,var))
    option.grid(row=0,column=0,sticky=(N,W))

我做了一些研究,我认为我做的一切都是正确的,但是当我在选项菜单中选择一个选项时,我得到了以下错误:

^{pr2}$

我想我已经把var传入了changeSchool函数。在

谢谢你的帮助!在

编辑:以下是我的changeSchool函数,供那些希望看到它的人使用:

def changeSchool(frame,school):
    print (school)
    WizTool.schoolType = school
    self.initBT(frame,WizTool.schoolType)
    print (WizTool.schoolType)

编辑:这是整个程序(显示框架如何不在范围内)

from tkinter import *
from tkinter import ttk
from functools import partial
import webbrowser
import pprint

class WizTool:
schoolType = "Fire"


#initialize the GUI
def __init__(self, master):
    content = ttk.Frame(master, padding=(3,3,12,12))
    frame = ttk.LabelFrame(content, borderwidth=5, relief="sunken", width=400, height=400,padding=(3,3,12,12),text = "Blade Tracker")
    content.grid(column=0,row=0,sticky=(N,S,E,W))
    frame.grid(column=0, row=0, columnspan=4, sticky=(N, S, E, W))
    self.master = master
    self.initUI(content)
    self.initBT(frame,WizTool.schoolType)


def initUI(self,frame):
    self.master.title("Bootstrapper")


def changeSchool(frame,school):
    print (school)
    WizTool.schoolType = school
    self.initBT(frame,WizTool.schoolType)
    print (WizTool.schoolType)

def initBT(self,frame,mySchool):
#option menu for selecting school

    OPTIONS = [
        "Fire",
        "Ice",
        "Storm",
        "Life",
        "Myth",
        "Death",
        "Balance"
    ]

    var = StringVar(frame)
    var.set("Select School") # initial value

    option = OptionMenu(frame, var,*OPTIONS,command= lambda frame,var : changeSchool(frame,var))
    option.grid(row=0,column=0,sticky=(N,W))





def main():

root = Tk()
root.geometry("800x500+300+300")
app = WizTool(root)
root.mainloop()

main()

Tags: importselfmastervardefcolumnframegrid
1条回答
网友
1楼 · 发布于 2024-06-16 11:53:23

OptionMenu小部件的command函数只接受一个参数:selected item。在

  1. 您正在调用command函数吗?回答:没有
  2. 谁在调用command函数?答案:Python。在

Python将您指定的函数存储在某个地方,然后在将来的某个时候,在options菜单中进行选择之后,Python调用command函数。因此python决定它将传递给command函数的参数数量,结果是python使用一个参数调用command函数:

 command(selection)
  1. 你怎么知道python用一个参数调用command函数?回答:你检查文件。

  2. 如果在文档中找不到描述您需要分配给command的函数类型的任何内容,会发生什么情况?回答:你来测试一下。

第一个:

^{pr2}$

下一步:

...
...
def do_stuff(x, y):
    print(x, y)

tk.OptionMenu(
    root, 
    str_var, 
    *OPTIONS,
    command=do_stuff).pack()

...
...

 output: 
> TypeError: do_stuff() missing 1 required positional argument: 'y'

有很多方法可以解决这个问题。。。在

使用python的作用域规则:

import tkinter as tk

OPTIONS = [
        "Fire",
        "Ice",
        "Storm",
        "Life",
        "Myth",
        "Death",
        "Balance"
]

root = tk.Tk()
root.title("Hello")
root.geometry("300x200+10+100")

frame = [1, 2, 3]

def do_stuff(selection):  
    print(selection, frame)  #frame is visible inside the def.

str_var = tk.StringVar(root)
str_var.set(OPTIONS[0]) # default value

tk.OptionMenu(
    root, 
    str_var, 
    *OPTIONS,
    command=do_stuff
).pack()

root.mainloop()

但是让函数操作全局变量并不是一个好主意,所以你可以。。。在

使用包装函数:

import tkinter as tk

OPTIONS = [
        "Fire",
        "Ice",
        "Storm",
        "Life",
        "Myth",
        "Death",
        "Balance"
]

root = tk.Tk()
root.title("Hello")
root.geometry("300x200+10+100")

def wrapper(other):
    def do_stuff(selection):
        print(selection, other)

    return do_stuff

str_var = tk.StringVar(root)
str_var.set(OPTIONS[0]) # default value

tk.OptionMenu(
    root, 
    str_var, 
    *OPTIONS,
    command=wrapper('xxxx')  #This actually executes wrapper(), and a
                             #function call in your code is replaced
                             #by its return value which happens to 
                             #be the name of a function that takes one
                             #argument.  The one arg function name is
                             #then assigned to command.
).pack()

root.mainloop()

使用额外参数变量的默认值:

frame = [1, 2, 3]

def do_stuff(selection, other=frame):  #<****HERE****
    print(selection, other)

str_var = tk.StringVar(root)
str_var.set(OPTIONS[0]) # default value

tk.OptionMenu(
    root, 
    str_var, 
    *OPTIONS,
    command=do_stuff
).pack()

警告:参数变量的默认值有自己的问题。定义函数时,默认值被赋值一次。随后,无论执行函数多少次,默认值都使用相同的值。这意味着,如果默认值是一个列表,并且第一次调用函数时更改了该列表,那么下次调用该函数时,默认值将是更改后的列表。在

相关问题 更多 >