【问题标题】:Why does a list of generators only return the elements of the last generator? [duplicate]为什么生成器列表只返回最后一个生成器的元素? [复制]
【发布时间】:2021-07-06 20:39:20
【问题描述】:

我得到了一个任意的对象列表(例如['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"] 的上下文中实际上是什么? 我需要如何调整我的代码以实现我最初的目标?

【问题讨论】:

  • 我注意到,如果我将my_iterators 更改为((element for _ in range(5)) for element in ["foo", "bar"]),它实际上会产生我预期的输出,但使用list((element for _ in range(5)) for element in ["foo", "bar"]) 它会再次产生我无法解释的输出。
  • 研究后期绑定.
  • 这是定义迭代器的更简单方法my_iterators = [[i]*5 for i in ['foo', 'bar']]
  • 这与我链接的 tkinter 副本中出现的问题基本相同,是后期绑定的结果。

标签: python python-3.x iterator generator


【解决方案1】:

正如 jonsharpe 和 David Buck 在 cmets 中指出的那样,实际上导致此处非直觉行为的是 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))

【讨论】:

    猜你喜欢
    • 2019-05-04
    • 2016-09-24
    • 2014-04-16
    • 1970-01-01
    • 2021-12-30
    • 1970-01-01
    • 1970-01-01
    • 2018-03-20
    • 1970-01-01
    相关资源
    最近更新 更多