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

Python 中 “yield“ 的不同行为

在我们使用Python编译过程中,yield 关键字用于定义生成器函数,它的作用是将函数变成一个生成器,可以迭代产生值。yield 的行为在不同的情况下会有不同的效果和用途。

在这里插入图片描述

1、问题背景

在 Python 中,“yield” 是一种生成器(generator)的实现方式。生成器是一种特殊类型的迭代器(iterator),它可以在运行时动态产生值。然而,在某些情况下,使用生成器可能会遇到令人困惑的行为。

比如,下面有一个函数 x(),它产生一个生成器,该生成器每次调用 next() 方法时都会递减全局变量 a 的值并产生一个 yield 语句:

a = 5def x():global aif a == 3:raise Exception("Stop")a = a - 1yield a

现在,让我们在 Python shell 中调用这个函数并打印出生成的值:

>>> print(x().next())
4
>>> print(x().next())
3

到目前为止,一切正常。但是,如果我们把生成器函数的调用结果赋值给一个变量,然后使用这个变量来产生值,就会出现不同的行为:

>>> a = 5
>>> b = x()
>>> print(b.next())
4
>>> b.next()
StopIteration

这次,在第二次调用 b.next() 时,它没有产生值,而是引发了一个 StopIteration 异常。这是为什么呢?

2、解决方案

要理解这种行为,我们需要了解生成器的工作原理。

当我们调用一个生成器函数时,它并不会立即执行函数体,而是返回一个生成器对象(generator object)。这个生成器对象包含了函数体中的代码,但它不会在调用时执行。当我们使用 next() 方法来产生值时,生成器对象才会开始执行函数体。

在第一次调用 x() 时,我们创建了一个新的生成器对象。这个对象在执行函数体时遇到了 a == 3 这个条件,并引发了一个异常。然后,我们在 Python shell 中打印出了这个异常。

在第二次调用 x() 时,我们又创建了一个新的生成器对象。这个对象在执行函数体时仍然遇到了 a == 3 这个条件,并引发了异常。

但是,当我们把生成器函数的调用结果赋值给变量 b 时,情况发生了变化。这使得我们可以多次调用 b.next() 来产生值。当我们第一次调用 b.next() 时,生成器对象从上次中断的地方继续执行,并产生了值 4

然而,当我们第二次调用 b.next() 时,生成器对象已经执行到了函数体的末尾,没有更多的值可以产生了。因此,它引发了一个 StopIteration 异常。

为了更好地理解这种行为,我们可以使用一个 for 循环来遍历生成器:

def looping(stop):for i in looping(stop):yield i>>> looping(3).next()
0
>>> looping(3).next()
0

注意,每次我们创建一个新的生成器,循环都会从头开始。然而,如果我们存储一个生成器的引用,那么循环会继续从上次中断的地方继续执行:

>>> stored = looping(3)
>>> stored.next()
0
>>> stored.next()
1
>>> stored.next()
2
>>> stored.next()
Traceback (most recent call last):File "<stdin>", line 1, in <module>
StopIteration

在循环期间,每次执行 yield 语句时,代码都会暂停;调用 .next() 继续从上一时间中断的地方继续执行函数。

StopIteration 异常是完全正常的;这是生成器传达它们已经完成的方式。一个 for 循环寻找这个异常来结束循环:

>>> for i in looping(3):
...     print(i)
...0
1
2

通过上述总结我们得知,yield 在不同的上下文中有不同的行为,但都涉及到生成器的创建或者协程的定义。所以说最终选择哪种模式还得更加自身情况来选择。

http://www.lryc.cn/news/344401.html

相关文章:

  • 迅睿CMS中实现关键词搜索高亮
  • 晶振的精度与稳定性有什么关系?
  • 【C】137 只出现一次的数字
  • 51单片机入门:DS1302时钟
  • Redis-5 分布式锁
  • 音转文工具,9.8k star! 【送源码】
  • 【首次发布】华为 OD 机试 C卷抽中题库清单(真题库),目前华为OD机考以C卷为主,特殊情况会发送D卷
  • 【进程等待】waitpid的参数pid | status的位图位操作WIFEXITEDWEXITSTATUS宏
  • unity---常用API
  • 设计模式: 模板模式
  • [虚拟机+单机]梦幻契约H5修复版_附GM工具
  • 头文件相互包含 前向声明
  • 七款好用的上网行为管理软件推荐 |有没有好用的上网行为管理系统
  • centos7-bcc 安装
  • 5.06号模拟前端面试8问
  • 解读Inscode AI:开启代码智能化的新时代
  • 快速了解Vuex
  • vue管理系统导航中添加新的iconfont的图标
  • Docker的介绍及与传统虚拟化技术的区别
  • 06.Git远程仓库
  • Anaconda安装和深度学习环境的安装(TensorFlow、Pytorch)
  • 元素设置 flex:1,但是会被内部长单词宽度超出拉伸
  • win11 安装oracle11g详细流程及问题总结
  • 自我模拟面试
  • 头歌java面向对象基础
  • PMP课程知识点很多,无法入手,该如何学习?
  • 隔离流量优化网络传输
  • 【前端热门框架【vue框架】】——事件处理与表单输入绑定以及学习技巧,让学习如此简单
  • 芒果YOLOv8改进164:检测头篇:ImplicitHead 隐性知识检测头| 即插即用,独家新颖更新,精度高效涨点
  • 学习周报:文献阅读+Fluent案例+有限体积法理论学习