将同步重写为异步:不等待函数

问题描述 投票:0回答:1

随着我读得越多,我对 python 中的 aysnc 感到更加愚蠢。所以我决定直接询问答案。如何更改以下代码(使用异步或类似方法)以获得所需的结果?另外,我怎样才能在烧瓶或sanic中做到这一点?

import time

def long_job():
    print('long job started')
    time.sleep(5)
    print('long job ended')

def main_job():
    long_job()
    time.sleep(1)
    print('main job returned')

main_job()

# expected result:
# 'long job started'
# 'main job returned'
# 'long job ended'

基本上,我不想在返回我的 main_job 之前等待 long_job 结束。先感谢您。 :)

python asynchronous sanic
1个回答
5
投票

等待 asyncio 的

sleep()
为其他工作腾出时间(如果您不需要等待其他事情)。 使用
create_task()
而不是
await
来启动作业而不阻塞。 最后,您必须使用事件循环开始主要工作。

# Written in Python 3.7
import asyncio

async def long_job():
    print('long job started')
    await asyncio.sleep(5)
    print('long job ended')

async def main_job():
    asyncio.create_task(long_job())
    await asyncio.sleep(1)
    print('main job returned')

你的框架应该启动事件循环;您不必自己启动它。您可以

await
或从框架调用的
create_task
函数中调用
main_job()
上的
async def
,具体取决于您是否要阻止。


如果您想在没有框架的情况下进行测试,则必须使用

asyncio.run()
自己启动循环。即使其他任务尚未完成,这也会在其任务完成后立即停止。但这很容易解决:

async def loop_job():
    asyncio.create_task(main_job())
    while len(asyncio.Task.all_tasks()) > 1:  # Any task besides loop_job()?
        await asyncio.sleep(0.2)

asyncio.run(loop_job())

如果您自己实现一个框架,您可以使用更原始的

loop.run_forever()
,但您必须自己
stop()

© www.soinside.com 2019 - 2024. All rights reserved.