async/await 是 python 对协程对象的显式标记和调度约定,async def 创建返回协程对象的函数,await 作为挂起点要求操作对象实现 await 协议并在事件循环中让出控制权。

async 与 await 不是线程,也不是协程的语法糖
它们是 Python 对 协程对象(coroutine object) 的显式标记和调度约定。当你写 async def func(),Python 实际生成的是一个返回协程对象的可调用对象;而 await 是一个**挂起点(suspend point)**,它要求被等待的对象必须是 awaitable(实现了 __await__ 方法),并在事件循环中让出控制权,而非阻塞线程。
底层依赖事件循环与状态机
每个协程函数编译后会生成一个 coroutine 对象,其内部是一个状态机(由 CPython 的 PyCoroObject 实现)。调用 await 时:
- 当前协程保存执行上下文(如局部变量、指令指针)并暂停;
- 控制权交还给事件循环(如
asyncio.run()启动的BaseEventLoop); - 事件循环检查其他就绪任务(I/O 完成、定时器触发、其他协程已唤醒),调度下一个可运行协程;
- 当被 await 的对象(如
asyncio.sleep()或loop.sock_recv())准备就绪,事件循环恢复原协程,从上次暂停处继续执行。
awaitable 对象必须满足协议
await 只能作用于符合 awaitable 协议 的对象,包括:
- 协程对象(
async def函数调用结果); - 实现了
__await__方法且返回迭代器的对象(如asyncio.Future); - 定义了
__await__并委托给另一个 awaitable 的自定义类(例如封装回调的异步句柄)。
若尝试 await 42 或 await list(),会抛出 TypeError: object int can't be used in 'await' expression —— 因为它们没实现该协议。
立即学习“Python免费学习笔记(深入)”;
async/await 与生成器有历史渊源但本质不同
早期 Python 用 @asyncio.coroutine + yield from 模拟协程,底层复用生成器机制。但 async/await(PEP 492)引入了专用字节码:GET_AWAITABLE、LOAD_CONST(空元组)、YIELD_FROM 被替换为更严格的 AWAIT 指令。这意味着:
- 协程对象不可被
next()或send()驱动; - 不能在普通函数中混用
yield和await; - 错误堆栈更清晰,调试时能直接定位到
await行而非生成器帧。








