生成器是协程调度的轻量载体,其对象封装代码对象、栈帧、指令偏移及执行上下文;yield 暂停保留局部变量,gi_frame.f_lasti 记录下条字节码;不可重入,StopIteration 后再调 next() 报 RuntimeError。
Python 生成器不是语法糖,是协程调度的轻量载体;不理解 yield 的状态机本质,就只能靠试错调 StopIteration。
调用含 yield 的函数时,Python 不执行函数体,而是立即返回一个 generator 对象。这个对象内部封装了:代码对象(__code__)、局部变量栈帧(gi_frame)、当前指令偏移(gi_running 和 gi_suspended 标志)、以及一个可恢复的执行上下文。
关键点:
gi_frame.f_lasti 记录下一条要执行的字节码位置,每次 next() 就从这里继续yield 暂停时,所有变量仍驻留在 gi_frame.f_locals 中StopIteration,gi_frame 被设为 None,再调 next() 必报 RuntimeError: generator already exhausted
yield 和 yield from 的行为差异yield 返回单个值并暂停;yield from 是委托协议,把子生成器的产出、异常传递和关闭信号全部代理出去。
常见误用场景:
yield from range(100000) 替代 for i in range(100000): yield i?可以,但注意:yield from 在 CPython 中有额外的帧切换开销,小数据量反而略慢yield from 会原样向上冒泡;而手动 for x in subgen(): yield x 会吞掉子生成器的 GeneratorExit
yield from 内部调用了 subgen().__iter__(),所以能委托任意可迭代对象(包括列表、文件对象),不只是生成器生成器函数中用 try...finally 可以保证退出时执行清理逻辑,但必须注意触发时机:
def safe_reader(filename):
f = open(filename)
try:
for line in f:
yield line.strip()
finally:
print("file closed")
f.close()
正常耗尽
gen = safe_reader("data.txt")
list(gen) # → 打印 "file closed"
提前中断
gen = safe_reader("data.txt")
next(gen)
gen.close() # → 打印 "file closed"
但这样不行:
gen = safe_re
ader("data.txt")
next(gen)
del gen # 不保证立刻触发 finally!依赖 GC,可能延迟甚至不触发
真正可靠的释放方式只有显式调用 close() 或用 with + 自定义上下文管理器包装生成器。
生成器天然适合建模状态流转:每个 yield 是一个稳定状态,send() 是外部输入事件。
def traffic_light():
state = "red"
while True:
cmd = yield state
if state == "red" and cmd == "go":
state = "green"
elif state == "green" and cmd == "stop":
state = "yellow"
elif state == "yellow" and cmd == "stop":
state = "red"
light = traffic_light()
next(light) # 启动,输出 "red"
print(light.send("go")) # → "green"
print(light.send("stop")) # → "yellow"
print(light.send("stop")) # → "red"
注意:send() 不能发给刚创建未启动生成器(需先 next() 或 send(None)),否则报 TypeError: can't send non-None value to a just-started generator。
这种写法容易忽略的是错误处理——没匹配的 cmd 会导致无限循环在当前状态,建议加默认分支抛异常或返回状态码。
# python
# go
# 字节
# 栈
# 状态码
# 可迭代对象
# red
# elif
相关文章:
如何快速搭建虚拟主机网站?新手必看指南
购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?
代购小票制作网站有哪些,购物小票的简要说明?
学校为何禁止电信移动建设网站?
建站之星安装后如何配置SEO及设计样式?
青岛网站设计制作公司,查询青岛招聘信息的网站有哪些?
天津个人网站制作公司,天津网约车驾驶员从业资格证官网?
如何构建满足综合性能需求的优质建站方案?
,怎么用自己头像做动态表情包?
网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?
常州企业建站如何选择最佳模板?
c# 在ASP.NET Core中管理和取消后台任务
高防服务器租用首荐平台,企业级优惠套餐快速部署
猪八戒网站制作视频,开发一个猪八戒网站,大约需要多少?或者自己请程序员,需要什么程序员,多少程序员能完成?
Android使用GridView实现日历的简单功能
如何在橙子建站上传落地页?操作指南详解
网站海报制作教学视频教程,有什么免费的高清可商用图片网站,用于海报设计?
专业制作网站的公司哪家好,建立一个公司网站的费用.有哪些部分,分别要多少钱?
招贴海报怎么做,什么是海报招贴?
如何通过主机屋免费建站教程十分钟搭建网站?
网站制作怎么样才能赚钱,用自己的电脑做服务器架设网站有什么利弊,能赚钱吗?
兔展官网 在线制作,怎样制作微信请帖?
青岛网站建设如何选择本地服务器?
内部网站制作流程,如何建立公司内部网站?
建站之星安全性能如何?防护体系能否抵御黑客入侵?
建站之星官网登录失败?如何快速解决?
简易网站制作视频教程,使用记事本编写一个简单的网页html文件?
建站主机选哪种环境更利于SEO优化?
微信h5制作网站有哪些,免费微信H5页面制作工具?
电脑免费海报制作网站推荐,招聘海报哪个网站多?
建站之星备案流程有哪些注意事项?
如何在景安云服务器上绑定域名并配置虚拟主机?
建站之星下载版如何获取与安装?
如何快速建站并高效导出源代码?
如何在万网自助建站中设置域名及备案?
如何在阿里云服务器自主搭建网站?
如何通过虚拟主机空间快速建站?
建站主机解析:虚拟主机配置与服务器选择指南
如何在七牛云存储上搭建网站并设置自定义域名?
如何生成腾讯云建站专用兑换码?
建站主机空间推荐 高性价比配置与快速部署方案解析
C++ static_cast和dynamic_cast区别_C++静态转换与动态类型安全转换
如何快速生成凡客建站的专业级图册?
高防网站服务器:DDoS防御与BGP线路的AI智能防护方案
南阳网站制作公司推荐,小学电子版试卷去哪里找资源好?
怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?
如何在云指建站中生成FTP站点?
C++如何使用std::optional?(处理可选值)
如何在IIS7中新建站点?详细步骤解析
网站制作新手教程,新手建设一个网站需要注意些什么?
*请认真填写需求信息,我们会在24小时内与您取得联系。