# 后台任务(Background tasks)

# 创建任务(Creating Tasks)

在异步 Python 里使用 任务 (opens new window) 是非常简单方便的,Sanic 提供了一种简单的方法来将任务添加到当前运行的 loop 当中, 它有点类似于 asyncio.create_task。关于在 "应用程序" loop 运行之前添加任务,请参见下一节。

async def notify_server_started_after_five_seconds():
    await asyncio.sleep(5)
    print('Server successfully started!')
app.add_task(notify_server_started_after_five_seconds())

Sanic 将尝试自动注入该应用程序,并将应用程序作为任务的参数进行传递。

async def auto_inject(app):
    await asyncio.sleep(5)
    print(app.name)
app.add_task(auto_inject)

或者您可以选择显式的将 app 作为传递参数。

async def explicit_inject(app):
    await asyncio.sleep(5)
    print(app.name)
app.add_task(explicit_inject(app))

#app.run 之前添加任务(Adding tasks before app.run)

在 app 运行前是可以添加后台任务的。为了完成这个目的,您应该为 Sanic.add_task 方法传入一个可调用的对象,即使用 async 定义的函数本身,而不是一个协程对象(调用 async 函数后得到的对象),当传入可调用对象之后,Sanic 将为 每一个 子程序都创建该任务。

注意:这样添加的任务是在 before_server_start 阶段工作的,因此在每个子进程都会有该任务(而不是在主进程中)。

这将会产生一些后果:请查看这个问题 (opens new window)这条评论 (opens new window)来获取更多信息。

要想只为主进程添加任务,您应该考虑使用 @app.main_process_start添加任务。注意:任务没完成子进程将无法启动。

示例代码:

async def slow_work(...):
   ...
app = Sanic(...)
app.add_task(slow_work) # Note: we are passing the callable and not coroutine object `slow_work(...)`
app.run(...)

提示

如果您想要为 slow_work 绑定一些参数,可以使用 functools.partial

# 命名任务(Named tasks)

这只适用于 Python3.8 及以上版本

当您创建任务的时候,您可以为您的任务指定一个名称,方便您后续进行任务追踪。

app.add_task(slow_work, name="slow_task")

之后,您可以使用 get_task 方法从应用程序的任何地方查看您的任务。

task = app.get_task("slow_task")

如果您想要取消任务,您可以通过 cacle_task 来进行操作,当然,该方法也是异步的,请确保使用时添加了 await

await app.cancel_task("slow_task")

所有注册的任务都可以在 app.tasks 属性中找到。为了防止已取消的任务填满,您可能需要运行 app.purge_tasks 来清除所有已完成或已取消的任务。

app.purge_tasks()

这种模式对于 websocket 来说非常有用:

async def receiver(ws):
    while True:
        message = await ws.recv()
        if not message:
            break
        print(f"Received: {message}")
@app.websocket("/feed")
async def feed(request, ws):
    task_name = f"receiver:{request.id}"
    request.app.add_task(receiver(ws), name=task_name)
    try:
        while True:
            await request.app.event("my.custom.event")
            await ws.send("A message")
    finally:
        # 当 websocket 关闭连接的时候,我们可以清除所有的任务。
        await request.app.cancel_task(task_name)
        request.app.purge_tasks()
MIT Licensed
Copyright © 2018-present Sanic Community Organization

~ Made with ❤️ and ☕️ ~