生成器是可暂停恢复的状态机:next()从上次yield继续执行,send(value)还传递value给上个yield;首次send必须为None;yield from实现协程双向通信;GeneratorExit触发finally清理;生成器表达式惰性求值省内存,列表推导式支持随机访问。
__next__ 和 send 到底怎么触发状态迁移生成器不是一次性执行完的函数,而是一个可暂停、可恢复的状态机。每次调用 __next__(或内置函数 next())时,它从上次 yield 暂停的位置继续执行,直到遇到下一个 yield 或函数结束。
而 send(value) 不仅会唤醒生成器,还会把 value 作为上一个 yield 表达式的返回值。注意:首次调用 send() 必须传 None,否则报 TypeError: can't send non-None value to a just-started generator。
next(gen) 或 gen.send(None),否则直接崩溃yield 左侧赋值(如 data = yield item)只在 send() 后生效;next() 相当于 send(None)
StopIteration),再调用任何方法都会引发 RuntimeError: generator already exhausted
def echo_gen():
while True:
received = yield "ready"
if received == "quit":
break
print(f"got: {received}")
g = echo_gen()
print(next(g)) # 输出 "ready"
print(g.send("hello")) # 输出 got: hello,返回 "ready"
g.send("quit") # 触发 break,下次 next 会 raise StopIteration
yield from 不只是语法糖,而是协程组合的关键原语yield from 不仅简化嵌套生成器的委托写法,更重要的是它建立了调用方、外层生成器、子生成器三者之间的双向通信通道——异常、return 值、send 数据都能穿透传递。
对比手写循环:for x in subgen: yield x 只能单向产出值,无法将外部 send 或 throw 透传给子生成器,也无法捕获子生成器的 return 值。
yield from subgen 会自动处理 StopIteration 并提取其 value 属性,作为当前表达式的返回值throw() 中断,外层生成器也会同步收到该异常(除非自己捕获)async/await 底层正是基于 yield from 构建的,所以理解它等于理解 await 的本质def sub():
yield 1
return "done"
def top():
result = yield from sub() # result = "done"
print(f"sub returned: {result}")
yield 2
g = top()
print(next(g)) # 1
print(next(g)) # 2
此时 sub 已 return,top 中 print 执行,然后抛出 StopIteration("done")

GeneratorExit 异常和 finally 块的真实执行时机当生成器对象被垃圾回收、或显式调用 close() 时,解释器会向其抛出 GeneratorExit 异常。这个异常不能被常规 except Exception: 捕获,必须显式写 except GeneratorExit:,且**禁止在该 except 块中 yield** —— 否则触发 RuntimeError。
更可靠的做法是把清理逻辑放在 finally 块里,它保证在生成器退出前(无论正常结束、close()、还是未捕获异常)都会执行。
GeneratorExit 是 BaseException 子类,不属于 Exception 体系,所以 except: 或 except Exception: 都抓不到finally 是最安全的资源释放位置,但要注意:若 finally 中发生未捕获异常,会覆盖原始的 GeneratorExit
close() 是唯一标准方式主动终止生成器并触发清理;不要依赖 GC 时间点def risky_gen():
try:
yield "working"
finally:
print("cleanup done") # close()、StopIteration、或异常退出时都会执行
g = risky_gen()
print(next(g))
g.close() # 输出 "cleanup done"
生成器表达式((x*2 for x in range(10**6)))和列表推导式([x*2 for x in range(10**6)])的核心差异不在语法,而在内存占用模型:前者是惰性求值的迭代器,后者是一次性分配并填充完整列表。
当数据量大、或下游只消费部分元素时,生成器明显胜出;但若需要多次遍历、随机索引、或长度判断(len()),就必须用列表——因为生成器只能单向消费一次,且没有长度属性。
len()、index()、切片([1:5])等操作list(gen) 会得到空列表(第二次已耗尽)真正该用生成器的场景:读大文件逐行处理、数据库游标流式获取、实时传感器数据管道、递归结构的深度优先遍历……这些都不是“省几MB内存”的问题,而是“不这么做就爆内存或阻塞”的刚需。
# python
# go
# ai
# 内存占用
# 为什么
相关文章:
如何快速搭建高效可靠的建站解决方案?
合肥制作网站的公司有哪些,合肥聚美网络科技有限公司介绍?
儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?
合肥做个网站多少钱,合肥本地有没有比较靠谱的交友平台?
网站制作员失业,怎样查看自己网站的注册者?
子杰智能建站系统|零代码开发与AI生成SEO优化指南
如何在Golang中指定模块版本_使用go.mod控制版本号
定制建站价位费用解析与套餐推荐全攻略
如何制作网站标识牌,动态网站如何制作(教程)?
如何在腾讯云服务器上快速搭建个人网站?
Android使用GridView实现日历的简单功能
平台云上自主建站:模板化设计与智能工具打造高效网站
小捣蛋自助建站系统:数据分析与安全设置双核驱动网站优化
如何高效利用亚马逊云主机搭建企业网站?
我的世界制作壁纸网站下载,手机怎么换我的世界壁纸?
建站与域名管理如何高效结合?
如何撰写建站申请书?关键要点有哪些?
如何登录建站主机?访问步骤全解析
如何在建站之星网店版论坛获取技术支持?
兔展官网 在线制作,怎样制作微信请帖?
成都网站制作价格表,现在成都广电的单独网络宽带有多少的,资费是什么情况呢?
如何通过西部数码建站助手快速创建专业网站?
北京制作网站的公司排名,北京三快科技有限公司是做什么?北京三快科技?
手机网站制作平台,手机靓号代理商怎么制作属于自己的手机靓号网站?
宝塔Windows建站如何避免显示默认IIS页面?
学校建站服务器如何选型才能满足性能需求?
武汉网站制作费用多少,在武汉武昌,建面100平方左右的房子,想装暖气片,费用大概是多少啊?
如何通过建站之星自助学习解决操作问题?
如何在Windows 2008云服务器安全搭建网站?
油猴 教程,油猴搜脚本为什么会网页无法显示?
公司门户网站制作公司有哪些,怎样使用wordpress制作一个企业网站?
如何在云主机上快速搭建多站点网站?
香港服务器网站卡顿?如何解决网络延迟与负载问题?
如何用虚拟主机快速搭建网站?详细步骤解析
如何在IIS中新建站点并配置端口与物理路径?
教程网站设计制作软件,怎么创建自己的一个网站?
C++如何将C风格字符串(char*)转换为std::string?(代码示例)
湖南网站制作公司,湖南上善若水科技有限公司做什么的?
如何通过商城免费建站系统源码自定义网站主题?
制作表格网站有哪些,线上表格怎么弄?
如何在局域网内绑定自建网站域名?
微信小程序制作网站有哪些,微信小程序需要做网站吗?
建站VPS推荐:2025年高性能服务器配置指南
如何选择高性价比服务器搭建个人网站?
建站之星官网登录失败?如何快速解决?
c++23 std::expected怎么用 c++优雅处理函数错误返回【详解】
网站制作话术技巧,网站推广做的好怎么话术?
免费制作海报的网站,哪位做平面的朋友告诉我用什么软件做海报比较好?ps还是cd还是ai这几个软件我都会些我是做网页的?
如何在阿里云虚拟机上搭建网站?步骤解析与避坑指南
如何在Golang中使用replace替换模块_指定本地或远程路径
*请认真填写需求信息,我们会在24小时内与您取得联系。