为什么生成器列表只有 return 最后一个生成器的元素?
Why does a list of generators only return the elements of the last generator?
我得到了一个任意的对象列表(例如 ['foo', 'bar']
)。我的目标是生成一个大小相等的列表,其中结果列表中的每个元素都是一个将相应输入元素重复 5 次的生成器。
这是对我实际想做的事情的一个很大的简化,我知道有很多方法可以解决这个任务。
但是,在我想如何解决这个问题时,我偶然发现了一些我无法解释的奇怪行为。
这是我对上述任务的解决方案:
my_iterators = [
(element for _ in range(5))
for element in ["foo", "bar"]
]
for my_iterator in my_iterators:
print(list(my_iterator))
我现在预计输出为:
['foo', 'foo', 'foo', 'foo', 'foo']
['bar', 'bar', 'bar', 'bar', 'bar']
然而,令我惊讶的是:
['bar', 'bar', 'bar', 'bar', 'bar']
['bar', 'bar', 'bar', 'bar', 'bar']
为什么 (element for _ in range(5))
似乎是 input_list 中最后一个元素的迭代器,而不管它在 for element in ["foo", "bar"]
的上下文中实际是什么?
我需要如何调整我的代码才能实现我最初的目标?
正如 jonsharpe 和 David Buck 在评论中指出的那样,真正导致此处非直觉行为的是 Python's late binding。
似乎有多种方法可以修复发布的代码,一种是:
def make_iterator(element):
return (element for _ in range(5))
my_iterators = [
make_iterator(element)
for element in ["foo", "bar"]
]
for my_iterator in my_iterators:
print(list(my_iterator))
我得到了一个任意的对象列表(例如 ['foo', 'bar']
)。我的目标是生成一个大小相等的列表,其中结果列表中的每个元素都是一个将相应输入元素重复 5 次的生成器。
这是对我实际想做的事情的一个很大的简化,我知道有很多方法可以解决这个任务。 但是,在我想如何解决这个问题时,我偶然发现了一些我无法解释的奇怪行为。
这是我对上述任务的解决方案:
my_iterators = [
(element for _ in range(5))
for element in ["foo", "bar"]
]
for my_iterator in my_iterators:
print(list(my_iterator))
我现在预计输出为:
['foo', 'foo', 'foo', 'foo', 'foo']
['bar', 'bar', 'bar', 'bar', 'bar']
然而,令我惊讶的是:
['bar', 'bar', 'bar', 'bar', 'bar']
['bar', 'bar', 'bar', 'bar', 'bar']
为什么 (element for _ in range(5))
似乎是 input_list 中最后一个元素的迭代器,而不管它在 for element in ["foo", "bar"]
的上下文中实际是什么?
我需要如何调整我的代码才能实现我最初的目标?
正如 jonsharpe 和 David Buck 在评论中指出的那样,真正导致此处非直觉行为的是 Python's late binding。
似乎有多种方法可以修复发布的代码,一种是:
def make_iterator(element):
return (element for _ in range(5))
my_iterators = [
make_iterator(element)
for element in ["foo", "bar"]
]
for my_iterator in my_iterators:
print(list(my_iterator))