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输出PowerPoint(ppt)文件中全部文字信息的方法
Apr 28 Python
理解python正则表达式
Jan 15 Python
Django 实现下载文件功能的示例
Mar 06 Python
对pycharm代码整体左移和右移缩进快捷键的介绍
Jul 16 Python
用pycharm开发django项目示例代码
Oct 24 Python
通过pykafka接收Kafka消息队列的方法
Dec 27 Python
python numpy 常用随机数的产生方法的实现
Aug 21 Python
Python多线程及其基本使用方法实例分析
Oct 29 Python
解决pycharm上的jupyter notebook端口被占用问题
Dec 17 Python
python实现简单猜单词游戏
Dec 24 Python
python缺失值的解决方法总结
Jun 09 Python
python使用torch随机初始化参数
Mar 22 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
IIS下配置Php+Mysql+zend的图文教程
2006/12/08 PHP
PHP array_multisort()函数的使用札记
2011/07/03 PHP
php删除与复制文件夹及其文件夹下所有文件的实现代码
2013/01/23 PHP
destoon在各个服务器下设置URL Rewrite(伪静态)的方法
2014/06/21 Servers
基于php的微信公众平台开发入门实例
2015/04/15 PHP
laravel 判断查询数据库返回值的例子
2019/10/11 PHP
js和jquery批量绑定事件传参数一(新猪猪原创)
2010/06/23 Javascript
JQuery+Ajax实现数据查询、排序和分页功能
2015/09/27 Javascript
js将table的每个td的内容自动赋值给其title属性的方法
2016/10/13 Javascript
NodeJS整合银联网关支付(DEMO)
2016/11/09 NodeJs
JavaScript数据结构之二叉树的计数算法示例
2017/04/13 Javascript
解决vue打包css文件中背景图片的路径问题
2018/09/03 Javascript
vue项目打包之后背景样式丢失的解决方案
2019/01/17 Javascript
详解单页面路由工程使用微信分享及二次分享解决方案
2019/02/22 Javascript
简单了解JavaScript中常见的反模式
2019/06/21 Javascript
vue监听用户输入和点击功能
2019/09/27 Javascript
Electron整合React使用搭建开发环境的步骤详解
2020/06/07 Javascript
Python pickle模块用法实例
2015/04/14 Python
python检查指定文件是否存在的方法
2015/07/06 Python
浅谈使用Python变量时要避免的3个错误
2017/10/30 Python
python学生信息管理系统(完整版)
2020/04/05 Python
python解析xml简单示例
2019/06/21 Python
简单了解python的内存管理机制
2019/07/08 Python
python 消费 kafka 数据教程
2019/12/21 Python
PyTorch的自适应池化Adaptive Pooling实例
2020/01/03 Python
python实现人机猜拳小游戏
2020/02/03 Python
django为Form生成的label标签添加class方式
2020/05/20 Python
日语专业推荐信
2013/11/12 职场文书
平面设计岗位职责
2013/12/14 职场文书
理工科学生的自我评价
2013/12/15 职场文书
黄河的主人教学反思
2014/02/07 职场文书
学雷锋活动总结报告
2014/06/26 职场文书
公务员党的群众路线教育实践活动学习心得体会
2014/10/30 职场文书
设立有限责任公司出资协议书
2014/11/01 职场文书
优秀班集体申报材料
2014/12/25 职场文书
2015军训通讯稿大全
2015/07/18 职场文书