2

我对以下代码的问题是通过 lambda 将“i”(只是一个简单的数字范围,但根据 number_boxes 更改)传递给回调,以便创建每个盒子的单独功能。

我已经尝试阅读教程并在我的代码中尝试了各种事情,但它要么不起作用,要么我得到错误'lambda()只需要 2 个参数,3 个给定'等。我相信我需要制作 'i' 一个列表但是我仍然得到这个特殊的错误..

我已经评论了出现问题的代码。我需要返回每个框中的值并覆盖文本。

谢谢你。

self.clicked = [] # In my __init__ definition
self.numbers = [StringVar() for i in xrange(self.number_boxes) ]   # Create Stringvar for each box

for i in xrange(self.number_boxes): # For each number, create a box

        self.clicked.append(False) # Not clicked in yet
        self.box.append(Entry(self.frame_table,bg='white',borderwidth=0, width=10, justify="center", textvariable=self.numbers[i], fg='grey')) # Textvariable where I enter a value to get after, need to do for each box (use box[i] but cannot for append)
        self.box[i].grid(row=row_list,column=column+i, sticky='nsew', padx=1, pady=1) 
        self.box[i].insert(0, "Value %g" % float(i+1))
        self.box[i].bind("<Button-1>", lambda event, index=i : self.callback(event, index)) # Need to pass the 'i's' to callback but having lambda difficulties

for i in self.numbers: 
        i.trace('w',lambda index=i: self.numberwritten(index) ) # Need for each box again here

def numberwritten(self, index): # A way of combining numberwritten and callback?
    po = self.box[index].get() # Get the values in each box
    print po

def callback(self, event, index):
        if (self.clicked[index] == False): # When clicked in box, overwrite default text with value and change colour to black, not grey
            self.box[index].delete(0, END)
            self.box[index].config(fg='black')
                self.clicked[index] = True

更新:当前问题:需要将“i”的所有值传递给回调,而不仅仅是一个,而是如何将列表放入 lambda?

错误:

Exception in Tkinter callback
Traceback (most recent call last):
File "C:\Python26\lib\lib-tk\Tkinter.py", line 1410, in __call__
return self.func(*args)
TypeError: lambda() takes at most 1 argument (3 given)
4

2 回答 2

2

您的代码非常接近,但是这一行:

self.box[i].bind("<Button-1>", lambda self, event,i : self.callback(event, data))

应该

self.box[i].bind("<Button-1>", lambda event, index=i: self.callback(event, index))

该对象index只是我们为其分配值的任意标签i。请注意,如果我们通过,Python 会抛出错误i,而不是变量名。而且我们不需要通过self;它通过在函数调用中的使用隐式传递:self.callback().

我唯一的其他评论是您应该clicked变成一个列表,以便您可以跟踪用户选择的对象。你可以完全按照形成的方式box形成它。祝你好运。


这里有一些关于clicked变成列表的提示,因为我认为这是您当前遇到的问题。编辑:根据 JF Sebastian 的评论更改了几行。

# Create an empty list based on the number of boxes created 
# above. This could go in your class's __init__ function, 
# or it could be declared globally, depending on the scope
# of self.box and the loop which determines its length.
self.clicked = ([False] * len(self.box))

# Finally, in your callback, check one member of the list,
# depending on which box was clicked.
def cb(self, event, index):
    if not self.clicked[index]:
        self.box[index].delete(0, END)
        self.box[index].config(fg='black')
        self.clicked[index] = True
    # Print the value of each of the widgets stored in the box list.
    for i, j in enumerate(self.box):
        print("%ith element: %s" % (i, j.get()))  
于 2012-07-06T01:06:05.830 回答
0

更新:看来您的 lambda 完全是多余的。完全删除它。

尝试:

self.box[i].bind("<Button-1>",self.callback)

代替:

lambda self, event, i: self.callback(event, i)

更新:

当我输入我的答案时,您似乎使情况恶化了,也许对 lambda-idiom 的一点澄清会比我上面的解决方案对您有更多帮助。;-)

lambda 函数也称为匿名函数,因为它们不需要名称。

比较以下示例:

>>>ulist = 'a1 s3 d2 f4 k6 w9'.split()

def normal_sort(L,R):
    result = int(L[1]) - int(R[1])
    return result

>>> sorted(ulist, normal_sort)
['a1', 'd2', 's3', 'f4', 'k6', 'w9']

与 lambda 函数相同:

sorted(ulist, lambda L, R: int(L[1])-int(R[1]))
['a1', 'd2', 's3', 'f4', 'k6', 'w9']

lambda函数有

  1. 无名
  2. 无参数括号
  3. 没有返回语句,因为它的单个语句的评估被作为返回值传回。

您可以通过将其分配给名称来命名 lambda 函数:

lambda_sort = lambda L, R: int(L[1])-int(R[1])

但是切换到正常功能可能已经很有用了。

于 2012-07-05T14:19:52.913 回答