Python yield的用法实例分析


Posted in Python onMarch 06, 2020

本文实例讲述了Python yield的用法。分享给大家供大家参考,具体如下:

yield的英文单词意思是生产,刚接触Python的时候感到非常困惑,一直没弄明白yield的用法。

只是粗略的知道yield可以用来为一个函数返回值塞数据,比如下面的例子:

def addlist(alist):
  for i in alist:
    yield i + 1

取出alist的每一项,然后把i + 1塞进去。然后通过调用取出每一项:

alist = [1, 2, 3, 4]
for x in addlist(alist):
  print x,

这的确是yield应用的一个例子

1.包含yield的函数

假如你看到某个函数包含了yield,这意味着这个函数已经是一个Generator,它的执行会和其他普通的函数有很多不同。比如下面的简单的函数:

def h():
  print 'To be brave'
  yield 5
h()

可以看到,调用h()之后,print 语句并没有执行!这就是yield,那么,如何让print 语句执行呢?这就是后面要讨论的问题,通过后面的讨论和学习,就会明白yield的工作原理了。

2.yield是一个表达式

Python2.5以前,yield是一个语句,但现在2.5中,yield是一个表达式(Expression),比如:

m = yield 5

表达式(yield 5)的返回值将赋值给m,所以,认为 m = 5 是错误的。那么如何获取(yield 5)的返回值呢?需要用到后面要介绍的send(msg)方法。

3.透过next()语句看原理

现在,我们来揭晓yield的工作原理。我们知道,我们上面的h()被调用后并没有执行,因为它有yield表达式,因此,我们通过next()语句让它执行。next()语句将恢复Generator执行,并直到下一个yield表达式处。比如:

def h():
  print 'Wen Chuan'
  yield 5
  print 'Fighting!'
c = h()
c.next()

c.next()调用后,h()开始执行,直到遇到yield 5,因此输出结果:

Wen Chuan

当我们再次调用c.next()时,会继续执行,直到找到下一个yield表达式。由于后面没有yield了,因此会??出异常:

Wen Chuan
Fighting!
Traceback (most recent call last):
  File "/home/evergreen/Codes/yidld.py", line 11, in <module>
    c.next()
StopIteration

4.send(msg) 与 next()

了解了next()如何让包含yield的函数执行后,我们再来看另外一个非常重要的函数send(msg)。其实next()和send()在一定意义上作用是相似的,区别是send()可以传递yield表达式的值进去,而next()不能传递特定的值,只能传递None进去。因此,我们可以看做

c.next() 和 c.send(None) 作用是一样的。

来看这个例子:

def h():
  print 'Wen Chuan',
  m = yield 5 # Fighting!
  print m
  d = yield 12
  print 'We are together!'
c = h()
c.next() #相当于c.send(None)
c.send('Fighting!') #(yield 5)表达式被赋予了'Fighting!'

输出的结果为:

Wen Chuan Fighting!

需要提醒的是,第一次调用时,请使用next()语句或是send(None),不能使用send发送一个非None的值,否则会出错的,因为没有yield语句来接收这个值。

5.send(msg) 与 next()的返回值

send(msg) 和 next()是有返回值的,它们的返回值很特殊,返回的是下一个yield表达式的参数。比如yield 5,则返回 5 。到这里,是不是明白了一些什么东西?本文第一个例子中,通过for i in alist 遍历 Generator,其实是每次都调用了alist.Next(),而每次alist.Next()的返回值正是yield的参数,即我们开始认为被压进去的东东。我们再延续上面的例子:

def h():
  print 'Wen Chuan',
  m = yield 5 # Fighting!
  print m
  d = yield 12
  print 'We are together!'
c = h()
m = c.next() #m 获取了yield 5 的参数值 5
d = c.send('Fighting!') #d 获取了yield 12 的参数值12
print 'We will never forget the date', m, '.', d

输出结果:

Wen Chuan Fighting!
We will never forget the date 5 . 12

6.throw() 与 close()中断 Generator

中断Generator是一个非常灵活的技巧,可以通过throw抛出一个GeneratorExit异常来终止Generator。Close()方法作用是一样的,其实内部它是调用了throw(GeneratorExit)的。我们看:

def close(self):
  try:
    self.throw(GeneratorExit)
  except (GeneratorExit, StopIteration):
    pass
  else:
    raise RuntimeError("generator ignored GeneratorExit")
# Other exceptions are not caught

因此,当我们调用了close()方法后,再调用next()或是send(msg)的话会抛出一个异常:

Traceback (most recent call last):
  File "/home/evergreen/Codes/yidld.py", line 14, in <module>
    d = c.send('Fighting!')  #d 获取了yield 12 的参数值12
StopIteration

更多关于Python相关内容可查看本站专题:《Python数据结构与算法教程》、《Python Socket编程技巧总结》、《Python函数使用技巧总结》、《Python字符串操作技巧汇总》及《Python入门与进阶经典教程》

希望本文所述对大家Python程序设计有所帮助。

Python 相关文章推荐
Python深入学习之闭包
Aug 31 Python
Python获取某一天是星期几的方法示例
Jan 17 Python
python安装oracle扩展及数据库连接方法
Feb 21 Python
详解Python中 sys.argv[]的用法简明解释
Dec 20 Python
Python cookbook(字符串与文本)在字符串的开头或结尾处进行文本匹配操作
Apr 20 Python
python numpy 显示图像阵列的实例
Jul 02 Python
python使用matplotlib画柱状图、散点图
Mar 18 Python
Django ORM 自定义 char 类型字段解析
Aug 09 Python
Django之使用celery和NGINX生成静态页面实现性能优化
Oct 08 Python
python为什么要安装到c盘
Jul 20 Python
python制作图形界面的2048游戏, 基于tkinter
Apr 06 Python
用Python写一个简易版弹球游戏
Apr 13 Python
用Python生成HTML表格的方法示例
Mar 06 #Python
使用Python第三方库pygame写个贪吃蛇小游戏
Mar 06 #Python
Python修改列表值问题解决方案
Mar 06 #Python
浅谈matplotlib.pyplot与axes的关系
Mar 06 #Python
python-xpath获取html文档的部分内容
Mar 06 #Python
关于python中的xpath解析定位
Mar 06 #Python
Python网络爬虫信息提取mooc代码实例
Mar 06 #Python
You might like
解决phpmyadmin 乱码,支持gb2312和utf-8
2006/11/20 PHP
PHP学习资料汇总与网址
2007/03/16 PHP
php实现阿拉伯数字和罗马数字相互转换的方法
2015/04/17 PHP
php生成txt文件实例代码介绍
2016/04/28 PHP
php用户登录之cookie信息安全分析
2016/05/13 PHP
laravel 中某一字段自增、自减的例子
2019/10/11 PHP
一段批量给页面上的控件赋值js
2010/06/19 Javascript
JavaScript中this的使用详解
2013/11/08 Javascript
JavaScript检查弹出窗口是否被阻拦的方法技巧
2015/03/13 Javascript
js跨域请求数据的3种常用的方法
2015/12/01 Javascript
第一次接触Bootstrap框架
2016/10/24 Javascript
web前端开发中常见的多列布局解决方案整理(一定要看)
2017/10/15 Javascript
详解vuex状态管理模式
2018/11/01 Javascript
JS+php后台实现文件上传功能详解
2019/03/02 Javascript
详解vue中使用axios对同一个接口连续请求导致返回数据混乱的问题
2019/11/06 Javascript
在vue中实现禁止屏幕滚动,禁止屏幕滑动
2020/07/22 Javascript
JS实现无限轮播无倒退效果
2020/09/21 Javascript
解决vue scoped html样式无效的问题
2020/10/24 Javascript
全面解读Python Web开发框架Django
2014/06/30 Python
分享6个隐藏的python功能
2017/12/07 Python
Django实现WebSSH操作物理机或虚拟机的方法
2019/11/06 Python
python GUI库图形界面开发之PyQt5线程类QThread详细使用方法
2020/02/26 Python
Python如何在bool函数中取值
2020/09/21 Python
matplotlib之pyplot模块坐标轴标签设置使用(xlabel()、ylabel())
2021/02/22 Python
北美个性化礼品商店:Things Remembered
2018/06/12 全球购物
CHARLES & KEITH英国官网:新加坡时尚品牌
2018/07/04 全球购物
俄罗斯宠物用品网上商店:ZooMag
2019/12/12 全球购物
blueseventy官网:铁人三项和比赛泳衣
2021/02/06 全球购物
丑小鸭教学反思
2014/02/03 职场文书
服务员岗位职责
2015/02/03 职场文书
文员岗位职责
2015/02/04 职场文书
2015年学校安全工作总结
2015/04/22 职场文书
pytorch MSELoss计算平均的实现方法
2021/05/12 Python
再次探讨go实现无限 buffer 的 channel方法
2021/06/13 Golang
小程序自定义轮播图圆点组件
2022/06/25 Javascript
云服务器部署 Web 项目的实现步骤
2022/06/28 Servers