带有 Lambda 值的字典更新所有条目

sem*_*ite 3 python lambda namedtuple python-2.7

我在 Python 2.7 中。我有两个类和一个命名元组。一个类包含一个字典作为实例属性和一个分配给该字典的函数。(这是情况的一个非常简化的版本)。namedtuple 很简单。另一个类是test_dict通过add_to_test_dict函数调用向其中添加条目的类。

然后我实例化 DictManipulator 并调用test函数:

from collections import namedtuple


class DictHolder(object):
    def __init__(self):
        self.test_dict = {}
    def add_to_test_dict(self, key, val):
        self.test_dict[key] = val

TestTuple = namedtuple('TestTuple', 'name data')

class DictManipulator(object):
    def test(self):
        named_tuple_list = [TestTuple(name='key1', data=1), TestTuple(name='key2', data=1000)]
        self.my_dh = DictHolder()
        for item in named_tuple_list:
            self.my_dh.add_to_test_dict(item.name, lambda: item.data)

my_dm = DictManipulator()
my_dm.test()
print('key1 value: ', my_dm.my_dh.test_dict['key1']())
print('key2 value: ', my_dm.my_dh.test_dict['key2']())
# ('key1 value: ', 1000)
# ('key2 value: ', 1000)
Run Code Online (Sandbox Code Playgroud)

为什么两个键在那里返回相同的值?我已经进行了足够的实验,可以说原始的 named_tuple_list 没有更新,我尝试使用lambda: copy.deepcopy(item.data),但这也不起作用。非常感谢,伙计们。

sch*_*ggl 6

这是一个典型的后期绑定问题(参见常见问题):当函数(作为 lambda/匿名与它无关)被调用时,它们访问 的当前值item,这是循环中的最后一个。尝试

lambda x=item: x.data 
Run Code Online (Sandbox Code Playgroud)

在你的循环中。这是有效的,因为默认参数在定义时绑定到函数,而在调用时评估公共局部变量。

类似(可能重复)的问题:Python Lambda in a loop

  • @semi.suite 是的,这是 Python 中的一个常见问题 :) 我链接了有关该主题的搭便车部分。 (3认同)