当前位置: 首页 > 面试题库 >

“Fire and forget” python async/await

养焱
2023-03-14
问题内容

有时需要进行一些非关键的异步操作,但我不想等待它完成。在Tornado的协程实现中,你可以通过简单地省略yield关键字来“触发并忘记”一个异步函数。

我一直在试图找出如何“火和忘记”与新的async/ await在Python 3.5发布的语法。例如,一个简化的代码片段:

async def async_foo():
    print("Do some stuff asynchronously here...")

def bar():
    async_foo()  # fire and forget "async_foo()"

bar()

但是,发生的事情是bar()永远不会执行,而是收到运行时警告:

RuntimeWarning: coroutine 'async_foo' was never awaited
  async_foo()  # fire and forget "async_foo()"

问题答案:

如果你使用的是Python> = 3.7,请在任何地方替换asyncio.ensure_futureasyncio.create_task最新的,更好的派生task的方法。

asyncio.Task到“激发并忘记”

根据python docsasyncio.Task说法,有可能启动一些协程以“在后台”执行。asyncio.ensure_future 函数创建的任务不会阻止执行(因此函数将立即返回!)。这似乎是你要求的一种“解雇”的方法。

import asyncio


async def async_foo():
    print("async_foo started")
    await asyncio.sleep(1)
    print("async_foo done")


async def main():
    asyncio.ensure_future(async_foo())  # fire and forget async_foo()

    # btw, you can also create tasks inside non-async funcs

    print('Do some actions 1')
    await asyncio.sleep(1)
    print('Do some actions 2')
    await asyncio.sleep(1)
    print('Do some actions 3')


if __name__ == '__main__':
    loop = asyncio.get_event_loop()
    loop.run_until_complete(main())

输出:

Do some actions 1
async_foo started
Do some actions 2
async_foo done
Do some actions 3

如果事件循环完成后正在执行任务怎么办?
请注意,asyncio期望任务在事件循环完成时完成。因此,如果你更改main()为:

async def main():
    asyncio.ensure_future(async_foo())  # fire and forget

    print('Do some actions 1')
    await asyncio.sleep(0.1)
    print('Do some actions 2')
程序完成后,你会收到以下警告:

Task was destroyed but it is pending!
task: <Task pending coro=<async_foo() running at [...]

为防止这种情况,你可以在事件循环完成后等待所有待处理的任务:

async def main():
    asyncio.ensure_future(async_foo())  # fire and forget

    print('Do some actions 1')
    await asyncio.sleep(0.1)
    print('Do some actions 2')


if __name__ == '__main__':
    loop = asyncio.get_event_loop()
    loop.run_until_complete(main())

    # Let's also finish all running tasks:
    pending = asyncio.Task.all_tasks()
    loop.run_until_complete(asyncio.gather(*pending))

杀死任务而不是等待任务

有时你不想等待任务完成(例如,某些任务可能创建为永久运行)。在这种情况下,你可以只取消()而不是等待它们:

import asyncio
from contextlib import suppress


async def echo_forever():
    while True:
        print("echo")
        await asyncio.sleep(1)


async def main():
    asyncio.ensure_future(echo_forever())  # fire and forget

    print('Do some actions 1')
    await asyncio.sleep(1)
    print('Do some actions 2')
    await asyncio.sleep(1)
    print('Do some actions 3')


if __name__ == '__main__':
    loop = asyncio.get_event_loop()
    loop.run_until_complete(main())

    # Let's also cancel all running tasks:
    pending = asyncio.Task.all_tasks()
    for task in pending:
        task.cancel()
        # Now we should await task to execute it's cancellation.
        # Cancelled task raises asyncio.CancelledError that we can suppress:
        with suppress(asyncio.CancelledError):
            loop.run_until_complete(task)

输出:

Do some actions 1
echo
Do some actions 2
echo
Do some actions 3
echo


 类似资料:
  • 问题内容: 我一直在使用“ if”来测试自己的版本,并且一切似乎都正常。当然,如果使用signalAll()而不是signal(),这将导致严重崩溃,但是如果一次仅通知一个线程,这怎么会出错? 他们的代码在这里 -检查put()和take()方法;在Condition的JavaDoc顶部可以看到一个更简单,更重点的实现。 下面是我实施的相关部分。 PS我知道,通常,尤其是在这样的lib类中,应该让

  • 问题内容: 我使用来执行任务。该任务可以递归创建提交给同一任务的其他任务,那些子任务也可以做到这一点。 我现在遇到的问题是,我要等到所有任务都完成(即所有任务都已完成并且它们没有提交新任务)后再继续。 我无法在主线程中调用,因为这会阻止接受新任务。 如果没有被呼叫,呼叫似乎无能为力。 所以我有点卡在这里。看到所有工人都闲着不难,不是吗?我能想到的唯一优雅的解决方案是直接使用a 并偶尔查询一次。真的

  • 问题内容: 我得到了asyncio在Python 3.5 中使用的流程,但是我还没有看到关于我应该使用什么东西,我不应该使用的await东西或者它在哪里容易出现的描述。我是否仅需要根据“这是IO操作并应进行await编辑” 来做出最好的判断? 问题答案: 默认情况下,所有代码都是同步的。你可以使用使其异步定义函数,async def并使用来“调用”这些函数await。一个更正确的问题是“什么时候应

  • 问题内容: 我正在使用该函数来控制一次最大操作数。 如您所见,我无法将该函数声明为异步的,因为我无权访问该函数的第二个回调中的值。 问题答案: 您实际上在promise构造函数执行程序函数内使用了promise,因此这是Promise构造函数anti-pattern。 您的代码很好地说明了主要风险:没有安全地传播所有错误。读为什么在那里。 此外,使用/ 会使相同的陷阱更加令人惊讶。相比: 天真(错

  • 问题内容: 给定下面的代码示例,行为上是否有任何差异,如果有,这些差异是什么? 据我了解,第一个将在异步函数中进行错误处理,并且错误将从异步函数的Promise中冒出来。但是,第二个需要较少的滴答声。它是否正确? 此片段只是返回Promise供参考的常用功能。 问题答案: 在大多数情况下,和之间没有可观察到的差异。这两个版本的观察行为完全相同(但根据实现的不同,该版本可能会使用更多的内存,因为可能

  • 问题内容: 以下函数从url获取图像并进行加载,然后返回其宽度和高度: 问题是,如果我做这样的事情: 我得到,因为该函数运行但图像尚未加载。 仅当照片已加载且宽度和高度已可用时,如何使用等待/异步返回值? 问题答案: 如何使用/ 将此回调函数转换为Promise? 你不知道与往常一样,您可以使用构造函数。没有语法糖。 仅当照片已加载且宽度和高度已经可用时,如何使用/ 记录值? 你可以做