0

0

Python await 背后到底发生什么

冰川箭仙

冰川箭仙

发布时间:2026-02-27 19:20:03

|

323人浏览过

|

来源于php中文网

原创

await 是协程暂停并交还控制权给事件循环的协作式调度机制,依赖 awaitable、事件循环和状态机协程三者配合;它非阻塞,本质是带语义的 yield,仅适用于 async def 函数。

python await 背后到底发生什么

当你写下 await,Python 并不是“暂停当前函数然后等结果回来”,而是把控制权交还给事件循环,让其他协程有机会运行——这背后是一套协作式调度机制,依赖于 可等待对象(awaitable)事件循环(event loop)状态机式的协程对象 三者配合。

await 不是阻塞,而是 yield 的升级版

一个 async def 函数被调用时,返回的是一个 协程对象(coroutine object),它本质上是一个可以被暂停和恢复的状态机。每次遇到 await,协程就暂停执行,并把自身(作为待恢复的“任务”)交还给事件循环。这和生成器中的 yield 行为高度相似,但语义更明确:你是在等待另一个 awaitable 完成。

  • 普通函数调用是同步栈式执行,会一直跑到 return 或异常;
  • await 触发的是协程的 __await__() 方法,它必须返回一个迭代器(通常是通过 yield 实现),事件循环靠这个迭代器一步步驱动协程前进;
  • 真正“挂起”的不是线程,而是当前协程的状态(局部变量、指令指针等),由 Python 解释器在 C 层面保存和恢复。

awaitable 对象必须满足什么条件?

只有符合 awaitable 协议的对象才能跟在 await 后面。Python 会按顺序检查:

  • 对象实现了 __await__ 方法,且返回一个迭代器(如 asyncio.Future、自定义类);
  • 对象本身是协程对象(async def 返回值);
  • 对象是 asyncio.Taskasyncio.Future 的实例(它们内部也实现了 __await__)。

如果写 await 42await "hello",Python 会立刻抛出 TypeError: object int can't be used in 'await' expression,因为这些类型没实现协议。

立即学习Python免费学习笔记(深入)”;

行者AI
行者AI

行者AI绘图创作,唤醒新的灵感,创造更多可能

下载

事件循环如何调度 await 的“等待”?

事件循环(如 asyncio.run() 启动的那个)并不真的“等”,而是维护一个就绪队列和一个等待队列:

  • 当协程 await 一个尚未完成的 Future,它会被注册到该 Future 的回调列表中;
  • Future 完成(比如 I/O 就绪、set_result() 被调用)时,会通知事件循环:“我好了”;
  • 事件循环把对应的协程重新加入就绪队列,在下一轮循环中继续驱动它执行到下一个 await 或结束。

整个过程不涉及线程切换或系统等待调用(除非底层用了 select/epoll 等 I/O 多路复用),纯用户态协作调度。

为什么不能在普通函数里用 await?

因为 await 是语法糖,只对 async def 定义的函数有效。它的存在本身就要求函数体被编译成一种特殊字节码(YIELD_FROM 指令),而普通函数没有这个能力。试图在非协程中写 await 会在语法解析阶段报错:SyntaxError: 'await' outside async function

如果你需要从同步上下文触发异步逻辑,必须显式进入事件循环,例如用 asyncio.run(coro())loop.run_until_complete(coro()) —— 这些函数才是真正启动调度器的地方。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

870

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

592

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

210

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

427

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

599

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

721

2023.08.10

function是什么
function是什么

function是函数的意思,是一段具有特定功能的可重复使用的代码块,是程序的基本组成单元之一,可以接受输入参数,执行特定的操作,并返回结果。本专题为大家提供function是什么的相关的文章、下载、课程内容,供大家免费下载体验。

494

2023.08.04

Golang 并发编程模型与工程实践:从语言特性到系统性能
Golang 并发编程模型与工程实践:从语言特性到系统性能

本专题系统讲解 Golang 并发编程模型,从语言级特性出发,深入理解 goroutine、channel 与调度机制。结合工程实践,分析并发设计模式、性能瓶颈与资源控制策略,帮助将并发能力有效转化为稳定、可扩展的系统性能优势。

2

2026.02.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.5万人学习

Django 教程
Django 教程

共28课时 | 4.6万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号