当前位置: 首页 > news >正文

Python 协程(终止协程和异常处理)

终止协程和异常处理

协程中未处理的异常会向上冒泡,传给 next 函数或 send 方法的调用
方(即触发协程的对象)。示例 16-7 举例说明如何使用示例 16-6 中由
装饰器定义的 averager 协程。

示例 16-7 未处理的异常会导致协程终止

>>> from coroaverager1 import averager
>>> coro_avg = averager()
>>> coro_avg.send(40) # ➊
40.0
>>> coro_avg.send(50)
45.0
>>> coro_avg.send('spam') # ➋
Traceback (most recent call last):
...
TypeError: unsupported operand type(s) for +=: 'float' and 'str'
>>> coro_avg.send(60) # ➌
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
StopIteration

❶ 使用 @coroutine 装饰器装饰的 averager 协程,可以立即开始发送
值。
❷ 发送的值不是数字,导致协程内部有异常抛出。
❸ 由于在协程内没有处理异常,协程会终止。如果试图重新激活协
程,会抛出 StopIteration 异常。
出错的原因是,发送给协程的 ‘spam’ 值不能加到 total 变量上。

示例 16-7 暗示了终止协程的一种方式:发送某个哨符值,让协程退
出。内置的 None 和 Ellipsis 等常量经常用作哨符值。Ellipsis 的
优点是,数据流中不太常有这个值。我还见过有人把 StopIteration
类(类本身,而不是实例,也不抛出)作为哨符值;也就是说,是像这
样使用的:my_coro.send(StopIteration)。

从 Python 2.5 开始,客户代码可以在生成器对象上调用两个方法,显式
地把异常发给协程。
这两个方法是 throw 和 close。
generator.throw(exc_type[, exc_value[, traceback]])
致使生成器在暂停的 yield 表达式处抛出指定的异常。如果生成
器处理了抛出的异常,代码会向前执行到下一个 yield 表达式,而产
出的值会成为调用 generator.throw 方法得到的返回值。如果生成器
没有处理抛出的异常,异常会向上冒泡,传到调用方的上下文中。

generator.close()

致使生成器在暂停的 yield 表达式处抛出 GeneratorExit 异常。
如果生成器没有处理这个异常,或者抛出了 StopIteration 异常(通
常是指运行到结尾),调用方不会报错。如果收到 GeneratorExit 异
常,生成器一定不能产出值,否则解释器会抛出 RuntimeError 异常。
生成器抛出的其他异常会向上冒泡,传给调用方。

下面举例说明如何使用 close 和 throw 方法控制协程。示例 16-8 列出
的是接下来的例子使用的 demo_exc_handling 函数。
示例 16-8 coro_exc_demo.py:学习在协程中处理异常的测试代码

class DemoException(Exception):
"""为这次演示定义的异常类型。"""def demo_exc_handling():print('-> coroutine started')while True:try:x = yieldexcept DemoException:print('*** DemoException handled. Continuing...')else:print('-> coroutine received: {!r}'.format(x))
raise RuntimeError('This line should never run.')

❶ 特别处理 DemoException 异常。
❷ 如果没有异常,那么显示接收到的值。
❸ 这一行永远不会执行。

示例 16-8 中的最后一行代码不会执行,因为只有未处理的异常才会中
止那个无限循环,而一旦出现未处理的异常,协程会立即终止。

demo_exc_handling 函数的常规用法如示例 16-9 所示。
示例 16-9 激活和关闭 demo_exc_handling,没有异常

>>> exc_coro = demo_exc_handling()
>>> next(exc_coro)
-> coroutine started
>>> exc_coro.send(11)
-> coroutine received: 11
>>> exc_coro.send(22)
-> coroutine received: 22
>>> exc_coro.close()
>>> from inspect import getgeneratorstate
>>> getgeneratorstate(exc_coro)
'GEN_CLOSED'

如果把 DemoException 异常传入 demo_exc_handling 协程,它会处
理,然后继续运行,如示例 16-10 所示。

示例 16-10 把 DemoException 异常传入 demo_exc_handling 不
会导致协程中止

>>> exc_coro = demo_exc_handling()
>>> next(exc_coro)
-> coroutine started
>>> exc_coro.send(11)
-> coroutine received: 11
>>> exc_coro.throw(DemoException)
*** DemoException handled. Continuing...
>>> getgeneratorstate(exc_coro)
'GEN_SUSPENDED'

但是,如果传入协程的异常没有处理,协程会停止,即状态变成
‘GEN_CLOSED’。示例 16-11 演示了这种情况。

示例 16-11 如果无法处理传入的异常,协程会终止

>>> exc_coro = demo_exc_handling()
>>> next(exc_coro)
-> coroutine started
>>> exc_coro.send(11)
-> coroutine received: 11
>>> exc_coro.throw(ZeroDivisionError)
Traceback (most recent call last):
...
ZeroDivisionError
>>> getgeneratorstate(exc_coro)
'GEN_CLOSED'

如果不管协程如何结束都想做些清理工作,要把协程定义体中相关的代
码放入 try/finally 块中,如示例 16-12。

示例 16-12 coro_finally_demo.py:使用 try/finally 块在协程终
止时执行操作

class DemoException(Exception):
"""为这次演示定义的异常类型。"""def demo_finally():print('-> coroutine started')try:while True:try:x = yieldexcept DemoException:print('*** DemoException handled. Continuing...')else:print('-> coroutine received: {!r}'.format(x))finally:print('-> coroutine ending')

Python 3.3 引入 yield from 结构的主要原因之一与把异常传入嵌套的
协程有关。另一个原因是让协程更方便地返回值。请继续往下读,了解
详情。

http://www.dtcms.com/a/277381.html

相关文章:

  • 晋升指南-笔记
  • 登录模块的静态登录
  • 基于Redis Streams的实时消息处理实战经验分享
  • 2025湖北省信息安全管理与评估赛项一阶段技能书
  • 当外卖骑手遇上“爽提学院”:一场关于专业的蜕变
  • 电商系统未来三年趋势:体验升级、技术赋能与模式重构
  • 海豚远程控制APP:随时随地,轻松掌控手机
  • 强化学习 (11)随机近似
  • 串口A和S的含义以及RT的含义
  • 大众点评商业模式:从内容护城河到竞争熔炉
  • MYSQL数据库----DCL语句
  • 初识JDBC的增删改
  • 12.3 安全内存区域划分
  • Oracle goldengate同步SQL server数据库测试实验中遇到的问题汇总
  • 股指期货的三种风险类型是什么?
  • 以太坊应用开发基础:从理论到实战的完整指南
  • 基于 STM32H743VIT6 的边缘 AI 实践:猫咪叫声分类 CNN 网络部署实战(已验证)中一些bug总结
  • 广东省省考备考(第四十四天7.13)——数量:数学运算(听课后强化训练)
  • IP 地址与网络基础全面解析
  • 飞算AI使用体验-一种基于项目工程思维的AI-Code思路
  • Web攻防-PHP反序列化魔术方法触发条件POP链构造变量属性修改黑白盒角度
  • iOS ish app 打印时间
  • 【Spring AOP】通知类型,@Pointcut、@Order(切面优先级)
  • 导入 SciPy 的 io 模块
  • CAPL报文信号接收和发送
  • Function CAll和MCP
  • 音视频学习(三十七):pts和dts
  • Web攻防-PHP反序列化原生内置类Exception类SoapClient类SimpleXMLElement
  • archive/tar: unknown file mode ?rwxr-xr-x
  • 数据结构 单链表(1)