我正在尝试使用标题'command'标记获得一个简单的ttk TreeView表,每个列排序,但它似乎不正常。我正在使用此问题的答案来实现功能:Tk treeview column sort
我的代码:
import tkinter as tk
from tkinter import ttk
def treeview_sort_column(tv, col, reverse):
print('sorting %s!' % col)
l = [(tv.set(k, col), k) for k in tv.get_children('')]
l.sort(reverse=reverse)
# rearrange items in sorted positions
for index, (val, k) in enumerate(l):
print('Moving Index:%r, Value:%r, k:%r' % (index, val, k))
tv.move(k, '', index)
# reverse sort next time
tv.heading(col, command=lambda: treeview_sort_column(tv, col, not reverse))
cols = ('name', 'path', 'time', 'pb')
root = tk.Tk()
root.geometry("700x500")
listbox = ttk.Treeview(root, columns=cols, show="headings")
for each in ('name', 'path', 'time','pb'):
listbox.heading(each,text=each.capitalize(),command=lambda: treeview_sort_column(listbox, each, False))
listbox.column( each, width=tk.font.Font().measure(each.title() ))
if not each == 'path':
listbox.column(each,stretch=False)
if not each == 'name':
listbox.column( each, anchor='center')
listbox.pack(expand=True, fill=tk.BOTH)
root.mainloop()
我遇到的问题是每次运行它时,它只按最后一列排序,而不是按你点击的列排序(通过treeview_sort_column函数中的print语句验证)。单击结果窗口中的任何列得到的输出:
sorting pb!
sorting pb!
sorting pb!
sorting pb!
sorting pb!
sorting pb!
sorting pb!
sorting pb!
如果我在for循环中将每个命令的动态创建更改为显式的,则在循环之后添加它,然后它按预期工作(即每个列自行排序)。
listbox.heading('name', command=lambda: treeview_sort_column(listbox, 'name', False))
listbox.heading('path', command=lambda: treeview_sort_column(listbox, 'path', False))
listbox.heading('time', command=lambda: treeview_sort_column(listbox, 'time', False))
listbox.heading('pb', command=lambda: treeview_sort_column(listbox, 'pb', False))
产生:
sorting name!
sorting name!
sorting path!
sorting path!
sorting time!
sorting time!
sorting pb!
sorting pb!
显然这是一个非常简单的例子,我的最终应用程序实际上将数据插入列中(也有更多列),但由于我甚至无法使这个简化版本工作,我很茫然。我的循环有什么问题导致lambda函数搞砸了?
我的系统:
答案 0 :(得分:8)
你应该以其他方式使用lambda。在你的代码中,每个lambda都获得一个指向同一个变量的指针,这就是为什么所有函数都做同样的事情。例如,您应该复制变量以使lambda唯一。如何做的一个例子:
for each in ('name', 'path', 'time','pb'):
listbox.heading(each,text=each.capitalize(),command=lambda each_=each: treeview_sort_column(listbox, each_, False))
简单示例:
funcs = []
for item in ('abc', 'def', 'ghi'):
funcs.append(lambda : print(item))
for f in funcs:
f()
打印:
ghi
ghi
ghi
但是固定lambda的版本:
funcs = []
for item in ('abc', 'def', 'ghi'):
funcs.append(lambda item_=item: print(item_))
for f in funcs:
f()
打印
abc
def
ghi