tkinter按钮表格

我想用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()

img

这个问题是因为在你创建每一个 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()


不知道你这个问题是否已经解决, 如果还没有解决的话:

如果你已经解决了该问题, 非常希望你能够分享一下解决方案, 写成博客, 将相关链接放在评论区, 以帮助更多的人 ^-^