asyncio任务取消本质是抛出CancelledError让协程主动退出;需捕获异常、释放资源、完成收尾;TaskGroup(Python 3.11+)自动统一取消并等待清理。
asyncio 中取消任务不是强制终止,而是通过抛出 CancelledError 让协程主动退出。真正“优雅”的关键在于:协程需捕获该异常、释放资源(如关闭连接、清理临时文件)、并完成必要的收尾逻辑。若任务内部阻塞在未响应取消的 IO(如某些同步库调用)或忽略异常,则无法真正取消。
TaskGroup 是目前最简洁、安全的方式——它自动绑定子任务生命周期,任一任务失败或被取消,其余任务会收到统一取消信号,并等待它们完成清理。
async with asyncio.TaskGroup() as tg: 创建上下文tg.create_task() 启动的任务共享同一个取消语义tg.cancel() 或上下文退出时异常传播,都会触发全部任务的取消流程示例:
import asyncioasync def worker(name: str): try: await asyncio.sleep(3) print(f"{name} 完成") except asyncio.CancelledError: print(f"{name} 被取消,正在清理...") await asyncio.sleep(0.5) # 模拟清理耗时 print(f"{name} 清理完毕") raise # 重新抛出,确保 TaskGroup 知道已处理
async def main(): try: async with asyncio.TaskGroup() as tg: tg.create_task(worker("A")) tg.create_task(worker("B")) tg.create_task(worker("C")) await asyncio.sleep(1) # 运行
1 秒后取消 tg.cancel() # 主动取消全部 except asyncio.CancelledError: print("主流程也被取消")
asyncio.run(main())
对于 Python
asyncio.Task 实例到列表.cancel()
await asyncio.gather(*tasks, return_exceptions=True) 等待全部完成(避免因某个任务抛 CancelledError 导致其他任务被中断)不要在协程中静默吞掉 CancelledError —— 若只写 except Exception: 或空 except:,取消信号会被吃掉,任务卡住。
慎用 time.sleep() / threading.Lock / requests.get() —— 这些是同步阻塞操作,不响应 asyncio 取消。应改用 asyncio.sleep()、异步锁(asyncio.Lock)、aiohttp 等原生异步库。
长时间计算循环需主动检测取消状态 —— 在循环体内定期调用 if asyncio.current_task().cancelled(): raise asyncio.CancelledError,防止 CPU 密集型任务无法响应。