一篇文章弄懂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标准库之itertools库的使用方法
Sep 07 Python
django框架自定义用户表操作示例
Aug 07 Python
Python面向对象之类和对象实例详解
Dec 10 Python
Django多数据库的实现过程详解
Aug 01 Python
Python logging设置和logger解析
Aug 28 Python
python中删除某个元素的方法解析
Nov 05 Python
python字符串的index和find的区别详解
Jun 20 Python
Tensorflow与Keras自适应使用显存方式
Jun 22 Python
keras:model.compile损失函数的用法
Jul 01 Python
Selenium webdriver添加cookie实现过程详解
Aug 12 Python
PyQt5多线程防卡死和多窗口用法的实现
Sep 15 Python
pycharm2020.2 配置使用的方法详解
Sep 16 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在Web开发领域的优势
2006/10/09 PHP
php判断字符串在另一个字符串位置的方法
2014/02/27 PHP
PHP编程开发怎么提高编程效率 提高PHP编程技术
2015/11/09 PHP
php轻松实现文件上传功能
2016/03/03 PHP
PHP使用Curl实现模拟登录及抓取数据功能示例
2018/04/27 PHP
PHP设计模式之简单工厂和工厂模式实例分析
2019/03/25 PHP
TP3.2.3框架文件上传操作实例详解
2020/01/23 PHP
this[] 指的是什么内容 讨论
2007/03/24 Javascript
jquery插件制作 表单验证实现代码
2012/08/17 Javascript
javascript计时器事件使用详解
2014/01/07 Javascript
快速学习AngularJs HTTP响应拦截器
2015/12/31 Javascript
javascript函数自动执行常用方法汇总
2016/03/28 Javascript
Bootstrap基本插件学习笔记之标签切换(17)
2016/12/08 Javascript
微信小程序开发之大转盘 仿天猫超市抽奖实例
2016/12/08 Javascript
Vue.js计算属性computed与watch(5)
2016/12/09 Javascript
jQuery插件FusionCharts绘制的2D条状图效果【附demo源码】
2017/05/13 jQuery
JavaScript实现开关等效果
2017/09/08 Javascript
AngularJS表单验证功能
2017/10/19 Javascript
node.js基于socket.io快速实现一个实时通讯应用
2019/04/23 Javascript
echarts饼图各个板块之间的空隙如何实现
2020/12/01 Javascript
[03:42]2014DOTA2西雅图国际邀请赛 Navi战队巡礼
2014/07/07 DOTA
python机器学习库常用汇总
2017/11/15 Python
python OpenCV学习笔记实现二维直方图
2018/02/08 Python
用 Django 开发一个 Python Web API的方法步骤
2020/12/03 Python
英国奢侈品网站:MatchesFashion
2016/12/16 全球购物
荷兰皇家航空公司官方网站:KLM Royal Dutch Airlines
2017/12/07 全球购物
草莓网化妆品澳大利亚站:Strawberrynet AU
2017/12/18 全球购物
比利时家具购买网站:Home24
2019/01/03 全球购物
澳大利亚拥有最好的家具和家居用品在线目的地:Nestz
2019/02/23 全球购物
基层干部2014全国两会学习心得体会
2014/03/10 职场文书
人力资源本科毕业生求职信
2014/06/04 职场文书
《中国梦我的梦》中学生演讲稿
2014/08/20 职场文书
2015年高中班主任工作总结
2015/04/30 职场文书
天气温馨提示语
2015/07/14 职场文书
初中运动会前导词
2015/07/20 职场文书
python 离散点图画法的实现
2022/04/01 Python