最新公告
  • 欢迎您光临网站无忧模板网,本站秉承服务宗旨 履行“站长”责任,销售只是起点 服务永无止境!立即加入钻石VIP
  • 深究Python中的asyncio库-shield函数

    正文概述    2020-07-20   276

    shield

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

    深究Python中的asyncio库-shield函数

    看一个例子:

    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库-函数的回调与调度


    下载网 » 深究Python中的asyncio库-shield函数

    常见问题FAQ

    免费下载或者VIP会员专享资源能否直接商用?
    本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
    提示下载完但解压或打开不了?
    最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,若小于网盘提示的容量则是这个原因。这是浏览器下载的bug,建议用百度网盘软件或迅雷下载。若排除这种情况,可在对应资源底部留言,或 联络我们.。
    找不到素材资源介绍文章里的示例图片?
    对于PPT,KEY,Mockups,APP,网页模版等类型的素材,文章内用于介绍的图片通常并不包含在对应可供下载素材包内。这些相关商业图片需另外购买,且本站不负责(也没有办法)找到出处。 同样地一些字体文件也是这种情况,但部分素材会在素材包内有一份字体下载链接清单。
    模板不会安装或需要功能定制以及二次开发?
    请QQ联系我们

    发表评论

    还没有评论,快来抢沙发吧!

    如需帝国cms功能定制以及二次开发请联系我们

    联系作者

    请选择支付方式

    ×
    迅虎支付宝
    迅虎微信
    支付宝当面付
    余额支付
    ×
    微信扫码支付 0 元