python - 如何理解 python 协程中的“yield from”?
问题描述
代码来自Fluent Python 第一版,
我无法理解 中的行while True:
,grouper
删除该行会引发StopIteration
错误。
但我发现grouper
没有while True:
那个工作的新版本。为什么group.send(None)
需要另一个循环while True:
(或另一个results[key] = yield from averager()
)?
我的理解是group.send(None)
将停止yield from averager()
并分配results[key]
一个值(Result(count, average)
)。就这样。
from collections import namedtuple
Result = namedtuple('Result', 'count average')
# the subgenerator
def averager(): # <1>
total = 0.0
count = 0
average = None
while True:
term = yield # <2>
if term is None: # <3>
break
total += term
count += 1
average = total/count
return Result(count, average) # <4>
# the delegating generator
def grouper(results, key): # <5>
while True: # <6>
results[key] = yield from averager() # <7>
# Another version works
#def grouper(results, key):
# results[key] = yield from averager()
# results[key] = yield from averager()
# the client code, a.k.a. the caller
def main(data): # <8>
results = {}
for key, values in data.items():
group = grouper(results, key) # <9>
next(group) # <10>
for value in values:
group.send(value) # <11>
group.send(None) # important! <12>
# print(results) # uncomment to debug
report(results)
# output report
def report(results):
for key, result in sorted(results.items()):
group, unit = key.split(';')
print('{:2} {:5} averaging {:.2f}{}'.format(
result.count, group, result.average, unit))
data = {
'girls;kg':
[40.9, 38.5, 44.3, 42.2, 45.2, 41.7, 44.5, 38.0, 40.6, 44.5],
'girls;m':
[1.6, 1.51, 1.4, 1.3, 1.41, 1.39, 1.33, 1.46, 1.45, 1.43],
'boys;kg':
[39.0, 40.8, 43.2, 40.8, 43.1, 38.6, 41.4, 40.6, 36.3],
'boys;m':
[1.38, 1.5, 1.32, 1.25, 1.37, 1.48, 1.25, 1.49, 1.46],
}
if __name__ == '__main__':
main(data)
解决方案
这让我想起了ascynio有多好,以及为什么每个人都应该使用它……
通过遍历迭代器的操作可以最好地解释正在发生的事情。这是内部生成器,简化:
def averager():
local_var
while True:
term = yield
if term is None:
break
local_var = do_stuff(term)
return local_var
这有两件事。首先,yield
只要这些数据不是None
. 然后当它是 None
时,它会提高 aStopIterationException
的值local_var
。(这就是从生成器返回的作用)。
这是外部生成器:
def grouper(results, key):
while True:
results[key] = yield from averager()
这样做是将内部生成器的 yield 暴露给调用代码,直到内部生成器 raise StopIterationException
,它被静默捕获(由yield from
语句)并分配。然后它准备好再次做同样的事情。
然后我们有调用代码:
def main(data):
results = {}
for key, values in data.items():
group = grouper(results, key)
next(group)
for value in values:
group.send(value)
group.send(None)
这是做什么的:
- 它只迭代外部生成器一次
- 这暴露了内部生成器的产量,并使用该 (
.send
) 与内部生成器进行通信。 - 它通过发送“结束”内部生成器
None
,此时第一yield from
条语句结束,并分配向上传递的值。 - 此时,外部生成器准备好发送另一个值
- 循环继续,生成器被垃圾回收删除。
while True:
循环是怎么回事?
考虑这段代码,它也适用于外部生成器:
def grouper(result, key):
result[key] = yield from averager
yield 7
唯一重要的是生成器不应该被耗尽,所以它不会在链上传递一个异常说“我没有什么可以迭代了”。
PS 糊涂了?我曾是。我不得不检查一下,自从我尝试使用基于生成器的 coros 以来已经有一段时间了。它们被安排删除——使用 asyncio,它更好。
推荐阅读
- c# - 从另一个泛型方法调用 MaybeNull 泛型方法
- javascript - 使用 Javascript 和 Webpack 加载图像数组?
- sql - REGEXP_LIKE、oracle、sql、查询、理解
- arm - 树莓派模型 3b+ 活动 LED 沿 gpio 4 闪烁
- mysql - 如何在没有行重复的情况下获得结果
- javascript - 正则表达式:匹配模式,除了前面的模式
- wpf - WPF按钮未触发单击事件
- c# - LINQ 当前记录和两边的 x 条记录
- django - NameError at "" name 'pk__in' 未定义
- javascript - xPath里面的xPath?