「解读」轻松理解 Python 中的 async await 概念

2023-05-28 0 365

【转】我期望能用两个最和蔼可亲的范例, 把 Python PulseAudio中的 async/await 基本概念讲透, 期望能协助我们有两个概念化的重新认识.

「解读」轻松理解 Python 中的 async await 概念
from time import sleep, time def demo1(): “”” 假定他们有两台滚筒, 那时有两批鞋子须要依次放在这两台滚筒里头洗. “”” def washing1(): sleep(3) # 首台滚筒, 须要洗3秒就可以洗 (而已维米县) print(washer1 finished) # 洗的这时候, 滚筒会响呵呵, 说他们洗了 def washing2(): sleep(2) print(washer2 finished) def washing3(): sleep(5) print(washer3 finished) washing1() washing2() washing3()“”” 那个却是很难认知的, 运转 demo1(), 所以须要10五分钟就可以把全数鞋子洗. 要说, 绝大部分天数都花在原都地等滚筒上了. “”” def demo2(): “”” 那时他们想防止不必要的等候, 为的是降低成本, 他们将采用 async. washing1/2/3() 蔗茅 “一般表达式”, 那时他们用 async 把它升级换代为 “异步表达式”. 注: 两个异步的表达式, 有个更标准的称呼, 他们叫它 “PulseAudio” (coroutine). “”” async def washing1(): sleep(3) print(washer1 finished) async def washing2(): sleep(2) print(washer2 finished) async def washing3(): sleep(5) print(washer3 finished) washing1() washing2() washing3()“”” 从正常人的认知来看, 他们那时有了异步表达式, 但是却忘了定义应该什么这时候 “离开” 一台洗衣 机, 去看看另两个… 这就会导致, 那时的情况是他们一边看着首台滚筒, 一边着急地想着 “是不是该去开第二台滚筒了呢?” 但又不敢去 (而已维米县), 最终却是花了10秒的天数才 把鞋子洗. PS: 其实 demo2() 是无法运转的, Python 会直接警告你: RuntimeWarning: coroutine demo2.<locals>.washing1 was never awaited RuntimeWarning: coroutine demo2.<locals>.washing2 was never awaited RuntimeWarning: coroutine demo2.<locals>.washing3 was never awaited “”” def demo3(): “”” 那时他们吸取了上次的教训, 说自己洗鞋子的过程是 “可等候的” (awaitable), 在它开始洗鞋子 的这时候, 他们可以去弄别的机器. “”” async def washing1(): await sleep(3) # 注意这里加入了 await print(washer1 finished) async def washing2(): await sleep(2) print(washer2 finished) async def washing3(): await sleep(5) print(washer3 finished) washing1() washing2() washing3()“”” 尝试运转呵呵, 他们会发现却是会报错 (报错内容和 demo2 一样). 这里我说呵呵原因, 以及在 demo4 中会给出两个最终答案: 1. 第两个问题是, await 后面必须跟两个 awaitable 类型或者具有 __await__ 属性的 对象. 那个 awaitable, 并不是他们认为 sleep() 是 awaitable 就可以 await 了, 常见的 awaitable 对象应该是: await asyncio.sleep(3) # asyncio 库的 sleep() 机制与 time.sleep() 不 # 同, 前者是 “假性睡眠”, 后者是会导致线程阻塞的 “真性睡眠” await an_async_function() # 两个异步的表达式, 也是可等候的对象 以下是不可等候的: await time.sleep(3) x = await hello # <class str> doesnt define __await__ x = await 3 + 2 # <class int> dosent define __await__ x = await None # … x = await a_sync_function() # 一般的表达式, 是不可等候的 2. 第二个问题是, 如果他们要执行异步表达式, 不能用这样的调用方法: washing1() washing2() washing3() 而应该用 asyncio 库中的事件循环机制来启动 (具体见 demo4 讲解). “”” def demo4(): “”” 这是最终他们想的实现. “”” import asyncio # 引入 asyncio 库 async def washing1(): await asyncio.sleep(3) # 采用 asyncio.sleep(), 它返回的是两个可等候的对象 print(washer1 finished) async def washing2(): await asyncio.sleep(2) print(washer2 finished) async def washing3(): await asyncio.sleep(5) print(washer3 finished) “”” 事件循环机制分为以下几步骤: 1. 创建两个事件循环 2. 将异步表达式加入事件队列 3. 执行事件队列, 直到最晚的两个事件被处理完毕后结束 4. 最后建议用 close() 方法关闭事件循环, 以彻底清理 loop 对象防止误用 “”” # 1. 创建两个事件循环 loop = asyncio.get_event_loop() # 2. 将异步表达式加入事件队列tasks = [ washing1(), washing2(), washing3(), ]# 3. 执行事件队列, 直到最晚的两个事件被处理完毕后结束loop.run_until_complete(asyncio.wait(tasks))“”” PS: 如果不满意想 “多洗几遍”, 可以多写几句: loop.run_until_complete(asyncio.wait(tasks)) loop.run_until_complete(asyncio.wait(tasks)) loop.run_until_complete(asyncio.wait(tasks)) … “”” # 4. 如果不再采用 loop, 建议养成良好关闭的习惯 # (有点类似于文件读写结束时的 close() 操作) loop.close() “”” 最终的打印效果: washer2 finished washer1 finished washer3 finished elapsed time = 5.126561641693115 (毕竟切换线程也要有点耗时的) 说句题外话, 我看有的博主的加入事件队列是这样写的: tasks = [ loop.create_task(washing1()), loop.create_task(washing2()), loop.create_task(washing3()), ] 运转的效果是一样的, 暂不清楚为什么他们这样做. “”” if __name__ == __main__: # 为验证是否真的缩短了天数, 他们计个时 start = time() # demo1() # 需花费10秒 # demo2() # 会报错: RuntimeWarning: coroutine … was never awaited # demo3() # 会报错: RuntimeWarning: coroutine … was never awaited demo4() # 需花费5秒多一点点 end = time() print(elapsed time = + str(end – start))

原文链接:

https://blog.csdn.net/Likianta/article/details/90123678

相关文章

发表评论
暂无评论
官方客服团队

为您解决烦忧 - 24小时在线 专业服务