1

我不确定以下代码中到底发生了什么:

def coroutine():
    lst = []
    try:
        while True:
            item = (yield lst)
            if item == 3:
                raise ValueError
            print('append {}'.format(item))
            lst.append(item)
    except GeneratorExit:
        print('GeneratorExit')

crt = coroutine()

next(crt)
print(crt.send(1))
print(crt.send(2))

try:
    print(crt.send(3))
except ValueError:
    pass

print(crt.send(4))

这输出:

append 1
[1]
append 2
[1, 2]

Traceback (most recent call last):
  File "D:\Documents and Settings\Brecht\Desktop\crt.py", line 25, in <module>
    print(crt.send(4))
StopIteration

当使用调试器单步执行代码时, on raise ValueError,执行跳转到except GeneratorExit:,但不执行此 except 子句的主体(不打印“GeneratorExit”)。为什么不?

除此之外,我不认为我可以在协程引发异常后以任何方式恢复协程?有什么特别的理由不允许这样做吗?这至少在我的特定用例中很有用:)

4

1 回答 1

4

当你抛出异常时,代码流总是被打断。一旦在其中抛出异常,就无法恢复中断的生成器。

来自PEP 342(通过增强生成器的协程)

与该next()方法一样,该send()方法返回生成器迭代器产生的下一个值,或者StopIteration在生成器正常退出或已经退出时引发。如果生成器引发未捕获的异常,则将其传播给send()'s 调用者。

至于调试器跳转到该except行:您只是抛出了一个异常,而解释器正在测试该异常是否被该行捕获。因为它只会捕获 a GeneratorExit,所以生成器会在此时退出。

于 2013-03-12T21:03:13.320 回答