国产成人AV无码一二三区,少女1到100集,国产精品久久久久精品综合紧,巜公妇之诱感肉欲HD在线播放

文章 > Python高级 > 深究Python中的asyncio库-shield函数

深究Python中的asyncio库-shield函数

Python

头像

Python

2019-06-04 11:05:405803浏览 · 0收藏 · 0评论

shield

asyncio.shield,用它可以屏蔽取消操作。一直到这里,我们还没有见识过Task的取消。

看一个例子:

In : loop = asyncio.get_event_loop()
In : task1 = loop.create_task(a())
In : task2 = loop.create_task(b())
In : task1.cancel()
Out: True
In : await asyncio.gather(task1, task2)
Suspending a
Suspending b
---------------------------------------------------------------------------
CancelledError                            Traceback (most recent call last)
cell_name in async-def-wrapper()
CancelledError:

在上面的例子中,task1被取消了后再用asyncio.gather收集结果,直接抛CancelledError错误了。这里有个细节,gather支持return_exceptions参数:

In : await asyncio.gather(task1, task2, return_exceptions=True)
Out: [concurrent.futures._base.CancelledError(), 'B']

可以看到,task2依然会执行完成,但是task1的返回值是一个CancelledError错误,也就是任务被取消了。如果一个创建后就不希望被任何情况取消,可以使用asyncio.shield?;と挝衲芩忱瓿?。不过要注意一个陷阱,先看错误的写法:

In : task1 = asyncio.shield(a())
In : task2 = loop.create_task(b())
In : task1.cancel()
Out: True
In : await asyncio.gather(task1, task2, return_exceptions=True)
Suspending a
Suspending b
Resuming b
Out: [concurrent.futures._base.CancelledError(), 'B']

可以看到依然是CancelledError错误,且协程a未执行完成,正确的用法是这样的:

In : task1 = asyncio.shield(a())
In : task2 = loop.create_task(b())
In : ts = asyncio.gather(task1, task2, return_exceptions=True)
In : task1.cancel()
Out: True
In : await ts
Suspending a
Suspending b
Resuming a
Resuming b
Out: [concurrent.futures._base.CancelledError(), 'B']

可以看到虽然结果是一个CancelledError错误,但是看输出能确认协程实际上是执行了的。所以正确步骤是:

先创建 GatheringFuture 对象 ts

取消任务

await ts

asynccontextmanager

如果你了解Python,之前可能听过或者用过contextmanager ,一个上下文管理器。通过一个计时的例子就理解它的作用:

from contextlib import contextmanager
async def a():
    await asyncio.sleep(3)
    return 'A'
async def b():
    await asyncio.sleep(1)
    return 'B'
async def s1():
    return await asyncio.gather(a(), b())
@contextmanager
def timed(func):
    start = time.perf_counter()
    yield asyncio.run(func())
    print(f'Cost: {time.perf_counter() - start}')

timed函数用了contextmanager装饰器,把协程的运行结果yield出来,执行结束后还计算了耗时:

In : from contextmanager import *
In : with timed(s1) as rv:
...:     print(f'Result: {rv}')
...:
Result: ['A', 'B']
Cost: 3.0052654459999992

大家先体会一下。在Python 3.7添加了asynccontextmanager,也就是异步版本的contextmanager,适合异步函数的执行,上例可以这么改:

@asynccontextmanager
async def async_timed(func):
    start = time.perf_counter()
    yield await func()
    print(f'Cost: {time.perf_counter() - start}')
async def main():
    async with async_timed(s1) as rv:
        print(f'Result: {rv}')
In : asyncio.run(main())
Result: ['A', 'B']
Cost: 3.00414147500004

async版本的with要用async with,另外要注意yield await func()这句,相当于yield + await func()

PS: contextmanager 和 asynccontextmanager 最好的理解方法是去看源码注释

下一节:深究Python中的asyncio库-函数的回调与调度

关注

关注公众号,随时随地在线学习

本教程部分素材来源于网络,版权问题联系站长!

日日AV夜夜添久久奶无码| 《爸开车我抱着妈妈去姥姥家》| 《父债女偿》在线观看全集| 100%透光裸妆| 顶到里面了~疼| 饭桌上张开腿给公| 成色P31S是国精产品吗 | 性高朝久久久久久久久久| 白日梦我 电视剧| 《师生恋:禁忌之恋》结局| 主动把乳头放进男朋友嘴巴里 | 日韩大片PPT免费PPT| 和妈妈做了怎么办心理咨询| 饭桌上张开腿给公| 久久精品国产亚洲AV四虎百花| 舌头伸进去添的我爽高潮| 人野兽马狗猪大全| 大乐透走势图表| 萧阳| 你对着摄像头C自己给我看| 机械师2| 妈妈我要亲亲你PPT| 久久久无码AV精品夜夜挺价格| 维密内衣秀| 美女和帅哥一起努力生产豆浆 | 地下室SMS录制的全部内容| 小SAO货撅起屁股扒开C打视频 | 交换第一次| 《高压监狱2》完整版| 《性爽2》电影| 《熟妇的荡欲》HD中字| 欧美人动物PPT免费模板大全| 校长揉捏陈若雪的奶头| EXO妈妈MV高清在线观看| 少女哔哩哔哩免费高清观看2| 免费网站在线观看大全电视剧| 美国禁忌结局1-4| 亚洲最大的综合性区域国际贸易 | 下面的小嘴又饿了| 老公不在家憋不住了该怎么办| 快一点慢一点上一点下一点歌曲