一篇文章弄懂Python中的可迭代对象、迭代器和生成器


Posted in Python onAugust 12, 2019

我们都知道,序列可以迭代。但是,你知道为什么吗? 本文来探讨一下迭代背后的原理。

序列可以迭代的原因:iter 函数。解释器需要迭代对象 x 时,会自动调用 iter(x)。内置的 iter 函数有以下作用:

(1) 检查对象是否实现了 iter 方法,如果实现了就调用它,获取一个迭代器。

(2) 如果没有实现 iter 方法,但是实现了 getitem 方法,而且其参数是从零开始的索引,Python 会创建一个迭代器,尝试按顺序(从索引 0 开始)获取元素。

(3) 如果前面两步都失败,Python 抛出 TypeError 异常,通常会提示“C objectis not iterable”(C 对象不可迭代),其中 C 是目标对象所属的类。

由此我们可以明确知道什么是 可迭代的对象: 使用 iter 内置函数可以获取迭代器的对象。即要么对象实现了能返回迭代器的 iter 方法,要么对象实现了 getitem 方法,而且其参数是从零开始的索引。

下面看一个实现了getitem方法的例子:

一篇文章弄懂Python中的可迭代对象、迭代器和生成器

输出结果:

一篇文章弄懂Python中的可迭代对象、迭代器和生成器

我们创建了一个类Eg1,并且为这个类实现了 getitem 方法, 它的实例化对象o1 就是可迭代对象。

下面我们看一个实现 iter 方法的例子,因为用到了迭代器,所以在此我们必须在明确一下迭代器的用法。 标准的迭代器接口有两个方法:

一篇文章弄懂Python中的可迭代对象、迭代器和生成器

返回下一个可用的元素,如果没有元素了,抛出 StopIteration异常。

一篇文章弄懂Python中的可迭代对象、迭代器和生成器

返回 self,以便在应该使用可迭代对象的地方使用迭代器,例如在 for 循环中。

一篇文章弄懂Python中的可迭代对象、迭代器和生成器

我们创建了Eg2类,并为它实现了 iter 方法,此方法返回一个迭代器Eg2Iterator。 Eg2Iterator 实现了我们之前所说的next和iter方法。 实例化对象,并循环输出:

一篇文章弄懂Python中的可迭代对象、迭代器和生成器

可见,和o1是一样的。

我们通过两种方法实现了一个自己的可迭代对象,再此过程中我们要明确可迭代的对象和迭代器之间的关系:

Python 从可迭代的对象中获取迭代器。

iter方法从我们自己创建的迭代器类中获取迭代器,而getitem方法是python内部自动创建迭代器。

至此,我们明白了如何正确地实现可迭代对象,并且引出了怎样实现迭代器,但是使用迭代器方法(即上面的例子2)的代码量有点大,下面我们来了解一下如何使用更符合 Python 习惯的方式实现 Eg2类。

一篇文章弄懂Python中的可迭代对象、迭代器和生成器

哦了!就这么简单优雅!不用再单独定义一个迭代器类!

这里我们使用了yield 关键字, 只要 Python 函数的定义体中有 yield 关键字,该函数就是生成器函数。调用生成器函数时,会返回一个生成器对象。也就是说,生成器函数是生成器工厂。 当然,例子3的代码还可以使用yield from进一步简化:

一篇文章弄懂Python中的可迭代对象、迭代器和生成器

到这里我们明白了 可迭代对象 和 迭代器,还引申出了生成器,但还有一点没有提,那就是生成器表达式。

使用生成器表达式例子4的代码可以修改为:

一篇文章弄懂Python中的可迭代对象、迭代器和生成器

在python中,所有生成器都是迭代器。

最后,总结一下:

(1)什么是可迭代对象? 可迭代对象要么实现了能返回迭代器的 iter 方法,要么实现了 getitem 方法而且其参数是从零开始的索引。

(2)什么是迭代器? 迭代器是这样的对象:实现了无参数的 next 方法,返回下一个元素,如果没有元素了,那么抛出 StopIteration 异常;并且实现iter 方法,返回迭代器本身。

(3)什么是生成器? 生成器是带有 yield 关键字的函数。调用生成器函数时,会返回一个生成器对象。

(4)什么是生成器表达式? 生成器表达式是创建生成器的简洁句法,这样无需先定义函数再调用。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
Python Trie树实现字典排序
Mar 28 Python
Python创建文件和追加文件内容实例
Oct 21 Python
简单的抓取淘宝图片的Python爬虫
Dec 25 Python
python实现通过代理服务器访问远程url的方法
Apr 29 Python
在Django的上下文中设置变量的方法
Jul 20 Python
浅谈Python爬取网页的编码处理
Nov 04 Python
python3获取两个日期之间所有日期,以及比较大小的实例
Apr 08 Python
基于scrapy的redis安装和配置方法
Jun 13 Python
Python爬虫使用浏览器cookies:browsercookie过程解析
Oct 22 Python
Python基于class()实现面向对象原理详解
Mar 26 Python
DjangoWeb使用Datatable进行后端分页的实现
May 18 Python
Python打包exe时各种异常处理方案总结
May 18 Python
Django认证系统实现的web页面实现代码
Aug 12 #Python
django 自定义过滤器(filter)处理较为复杂的变量方法
Aug 12 #Python
django-filter和普通查询的例子
Aug 12 #Python
利用python实现汉字转拼音的2种方法
Aug 12 #Python
python面向对象 反射原理解析
Aug 12 #Python
Python中正反斜杠(‘/’和‘\’)的意义与用法
Aug 12 #Python
Django 查询数据库并返回页面的例子
Aug 12 #Python
You might like
在PHP中使用Sockets 从Usenet中获取文件
2008/01/10 PHP
PHP同时连接多个mysql数据库示例代码
2014/03/17 PHP
php页面,mysql数据库转utf-8乱码,utf-8编码问题总结
2015/08/27 PHP
PHP token验证生成原理实例分析
2019/06/05 PHP
TNC vs RR BO3 第一场 2.14
2021/03/10 DOTA
JS类中定义原型方法的两种实现的区别
2007/03/08 Javascript
直接生成打开窗口代码,不必下载
2008/05/14 Javascript
javascript字符串拼接的效率问题
2010/12/25 Javascript
javascript复制对象使用说明
2011/06/28 Javascript
js/jquery解析json和数组格式的方法详解
2014/01/09 Javascript
JQuery实现网页右侧随动广告特效
2016/01/17 Javascript
jQuery实现点击后高亮背景固定显示的菜单效果【附demo源码下载】
2016/09/21 Javascript
微信小程序之前台循环数据绑定
2017/08/18 Javascript
Vue下路由History模式打包后页面空白的解决方法
2018/06/29 Javascript
webpack@v4升级踩坑(小结)
2018/10/08 Javascript
vue实现搜索过滤效果
2019/05/28 Javascript
基于canvas实现手写签名(vue)
2020/05/21 Javascript
利用TensorFlow训练简单的二分类神经网络模型的方法
2018/03/05 Python
解决DataFrame排序sort的问题
2018/06/07 Python
python实现随机梯度下降法
2020/03/24 Python
python使用time、datetime返回工作日列表实例代码
2019/05/09 Python
在pycharm中显示python画的图方法
2019/08/31 Python
浅谈python的elementtree模块处理中文注意事项
2020/03/06 Python
django 解决扩展自带User表遇到的问题
2020/05/14 Python
HTML5+CSS3实例 :canvas 模拟实现电子彩票刮刮乐代码
2016/12/30 HTML / CSS
联想澳大利亚官网:Lenovo Australia
2018/01/18 全球购物
道德模范先进事迹
2014/02/14 职场文书
战友聚会主持词
2014/04/02 职场文书
大学应届毕业生求职信
2014/05/24 职场文书
组织鉴定材料
2014/06/02 职场文书
师范毕业生求职信
2014/07/11 职场文书
介绍信的格式
2015/01/30 职场文书
就业意向书范本
2015/05/11 职场文书
2019年朋友圈经典励志语录50条
2019/07/05 职场文书
关于考试抄袭的检讨书
2019/11/02 职场文书
LayUI+Shiro实现动态菜单并记住菜单收展的示例
2021/05/06 Javascript