当前位置: 首页 > java >正文

python函数之间嵌套使用yield

假设一种场景,函数 A 可以在获得函数 B 的返回值(即一个生成器对象)后,再次对其进行 yield 操作。这是因为 Python 的生成器是可迭代的,你可以在一个生成器中迭代另一个生成器,并将其结果逐个 yield 出去。

这种模式在 Python 中被称为 “生成器委托”(Generator Delegation),并且可以使用 yield from 语法来简化代码。


示例代码

方法 1:手动迭代并 yield

你可以在函数 A 中手动迭代函数 B 返回的生成器,并逐个 yield 其值:

def function_B():yield "Hello"yield "World"def function_A():generator_B = function_B()  # 调用函数 B,得到一个生成器for value in generator_B:yield value  # 逐个 yield 函数 B 的值# 测试
for item in function_A():print(item)

输出:

Hello
World
方法 2:使用 yield from(推荐)

Python 提供了 yield from 语法,可以直接将一个生成器的结果委托给另一个生成器,从而避免手动迭代:

def function_B():yield "Hello"yield "World"def function_A():yield from function_B()  # 使用 yield from 直接委托给函数 B# 测试
for item in function_A():print(item)

输出:

Hello
World

工作原理

  1. 生成器的本质

    • 当调用 function_B() 时,它不会立即执行函数体,而是返回一个生成器对象。
    • 生成器对象是一个可迭代对象,每次调用 next() 时会执行到下一个 yield 语句。
  2. yield from 的作用

    • yield from 是一种语法糖,用于将一个生成器的结果直接委托给另一个生成器。
    • 它会自动处理生成器的迭代过程,并将值逐个传递给外部调用者。
  3. 嵌套生成器

    • 在函数 A 中,你可以嵌套多个生成器,并通过 yieldyield from 将它们的结果逐层传递。

更复杂的例子

嵌套多个生成器

假设我们有多个生成器函数,并且希望在一个顶层生成器中组合它们的结果:

def generator_1():yield "A"yield "B"def generator_2():yield "C"yield "D"def combined_generator():yield from generator_1()  # 委托给 generator_1yield from generator_2()  # 委托给 generator_2# 测试
for item in combined_generator():print(item)

输出:

A
B
C
D
在中间插入额外逻辑

你还可以在 yield from 的基础上插入额外的逻辑:

def generator_1():yield "A"yield "B"def generator_2():yield "C"yield "D"def combined_generator():yield "Start"yield from generator_1()yield "Middle"yield from generator_2()yield "End"# 测试
for item in combined_generator():print(item)

输出:

Start
A
B
Middle
C
D
End

注意事项

  1. 生成器的状态

    • 生成器是有状态的,一旦迭代完成,就不能再次迭代。如果需要重新迭代,必须重新调用生成器函数。
  2. 异常处理

    • 如果被委托的生成器抛出异常,异常会传播到外层生成器。你可以使用 try...except 来捕获和处理异常。
  3. 性能优化

    • 使用 yield from 比手动迭代更高效,因为它减少了中间层的开销。
http://www.xdnf.cn/news/526.html

相关文章:

  • langchain langgraph 快速集成mcp: langchain-mcp-adapters
  • 历史文化探险,梧州旅游景点推荐
  • 任意文字+即梦3.0的海报设计Prompt
  • 基于尚硅谷FreeRTOS视频笔记——15—系统配制文件说明与数据规范
  • 基于MCP的RAG系统实战:用Cursor+GroundX构建复杂文档问答引擎
  • Java Spring Bean生命周期详解
  • AI 驱动抗生素发现:从靶点到化合物测试
  • 功能安全实战系列07-英飞凌TC3xx电源监控开发详解
  • 26考研——存储系统_主存储器与 CPU 的连接(3)
  • CUDA编程中影响性能的小细节总结
  • 《关于加快推进虚拟电厂发展的指导意见》解读
  • 图像预处理-图像边缘检测(流程)
  • OSI七层网络模型详解
  • Datawhale AI春训营】AI + 新能源(发电功率预测)Task1
  • 【KWDB创作者计划】_从0到1部署KWDB:踩坑指南与最佳实践
  • 深入理解 MCP 协议:开启 AI 交互新时代
  • Django 实现服务器主动给客户端发送消息的几种常见方式及其区别
  • 机器学习误差图绘
  • [HOT 100] 1964. 找出到每个位置为止最长的有效障碍赛跑路线
  • PHP中stdClass详解
  • 【java实现+4种变体完整例子】排序算法中【计数排序】的详细解析,包含基础实现、常见变体的完整代码示例,以及各变体的对比表格
  • 接口自动化 ——fixture allure
  • PHP异常处理__Throwable
  • STM32单片机入门学习——第42节: [12-2] BKP备份寄存器RTC实时时钟
  • Unity:获取组件对象(GetComponent<T>())
  • 栈(c++)
  • 单例模式:懒汉式的两种优化写法
  • Unity webgl 获取图片或视频数据
  • 【unity】Vulkan模式下部分Android机型使用VideoPlayer组件播放视频异常问题
  • 交易系统的构建与实战法则