我想用tkinter建一个144个按钮12*12的表格,每个按钮的初始样式是raised,按下后变成groove,但是不管按下哪个按钮都是最后一个按钮(11,11)变化,怎么解决
from tkinter import *
tk=Tk()
tk.geometry('600x600') #建窗口
l=[[None for i in range(12)] for j in range(12)] #按钮列表
press=(0,0) #当前按下的按钮位置
def f(i,j): #按下某个按钮
global press
l[press[0]][press[1]].config(relief='raised') #恢复上次按下的按钮
press=(i,j)
l[i][j].config(relief='groove') #改变当前按下按钮的样式
for i in range(12):
for j in range(12):
l[i][j]=Button(tk,text=' ',relief='raised',command=lambda: f(i,j)) #创建12*12的按钮
l[i][j].grid(row=i,column=j)
tk.mainloop()
这个问题是因为在你创建每一个 Button 时,都使用了同一个 lambda 表达式作为按钮的 command。lambda 表达式中的 i 和 j 是引用变量,当最后一个按钮创建完成后,它们的值已经被更新成了最后一个按钮的位置。
要解决这个问题,可以将 lambda 表达式改为一个普通函数,并将该函数作为参数传递给 command。例如:
def on_button_click(i, j):
global press
l[press[0]][press[1]].config(relief='raised') # 恢复上次按下的按钮
press = (i, j)
l[i][j].config(relief='groove') # 改变当前按下按钮的样式
for i in range(12):
for j in range(12):
l[i][j] = Button(tk, text=' ', relief='raised', command=lambda i=i, j=j: on_button_click(i, j))
l[i][j].grid(row=i, column=j)
在这个版本的代码中,我们定义了一个名为 on_button_click 的函数来处理按钮被按下的事件。该函数接受两个参数 i 和 j,表示当前被按下的按钮的位置。在创建 Button 对象时,我们将 i 和 j 的值传递给 lambda 表达式,以便它们能够正确地绑定到对应的按钮上。
这样做之后,应该就能够正常记录当前被按下的按钮的位置了。
你对闭包没有正确理解
command=lambda: f(i,j) 这里引用的i j是局部变量,在函数执行完后都是11 11,正确做法参考:https://www.didiok.com/article/8017
看其中的 使用闭包为多个 button 增加事件处理函数
lambda 函数捕获的是循环变量的值,而不是循环变量本身,导致每个按钮的回调函数都引用了同一个变量。解决方法是使用默认参数或者 functools.partial() 来传递循环变量的值。
默认参数解决方法:
from tkinter import *
tk = Tk()
tk.geometry('600x600')
l = [[None for i in range(12)] for j in range(12)]
press = (0, 0)
def f(i, j, btn=None): # 添加一个默认参数
global press
l[press[0]][press[1]].config(relief='raised')
press = (i, j)
btn.config(relief='groove') # 使用传入的按钮对象
for i in range(12):
for j in range(12):
l[i][j] = Button(tk, text=' ', relief='raised', command=lambda i=i, j=j: f(i, j, l[i][j])) # 传入按钮对象
l[i][j].grid(row=i, column=j)
tk.mainloop()
用 functools.partial() 的解决方法:
from tkinter import *
from functools import partial
tk = Tk()
tk.geometry('600x600')
l = [[None for i in range(12)] for j in range(12)]
press = (0, 0)
def f(i, j, btn):
global press
l[press[0]][press[1]].config(relief='raised')
press = (i, j)
btn.config(relief='groove')
for i in range(12):
for j in range(12):
l[i][j] = Button(tk, text=' ', relief='raised', command=partial(f, i, j, l[i][j]))
l[i][j].grid(row=i, column=j)
tk.mainloop()
不知道你这个问题是否已经解决, 如果还没有解决的话: