一篇文章弄懂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 相关文章推荐
Python2包含中文报错的解决方法
Jul 09 Python
python pyheatmap包绘制热力图
Nov 09 Python
python 运用Django 开发后台接口的实例
Dec 11 Python
Python判断一个list中是否包含另一个list全部元素的方法分析
Dec 24 Python
Python判断telnet通不通的实例
Jan 26 Python
详解Python 定时框架 Apscheduler原理及安装过程
Jun 14 Python
解决python文件双击运行秒退的问题
Jun 24 Python
Django MEDIA的配置及用法详解
Jul 25 Python
Python 通过截图匹配原图中的位置(opencv)实例
Aug 27 Python
springboot配置文件抽离 git管理统 配置中心详解
Sep 02 Python
python实现图像全景拼接
Mar 27 Python
基于Python爬取素材网站音频文件
Oct 21 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计数器的实现代码
2013/06/08 PHP
将PHP程序中返回的JSON格式数据用gzip压缩输出的方法
2016/03/03 PHP
在 Laravel 中 “规范” 的开发短信验证码发送功能
2017/10/26 PHP
Laravel框架实现修改登录和注册接口数据返回格式的方法
2018/08/17 PHP
PDO::quote讲解
2019/01/29 PHP
Laravel 手动开关 Eloquent 修改器的操作方法
2019/12/30 PHP
禁止刷新,回退的JS
2006/11/25 Javascript
jQuery chili图片远处放大插件
2009/11/30 Javascript
ext 列表页面关于多行查询的办法
2010/03/25 Javascript
js中跨域方法原理详解
2015/07/19 Javascript
js和jquery分别验证单选框、复选框、下拉框
2015/12/17 Javascript
jQuery选择器实例应用
2017/01/05 Javascript
原生Javascript插件开发实践
2017/01/18 Javascript
Swiper实现轮播图效果
2017/07/03 Javascript
微信小程序之蓝牙的链接
2017/09/26 Javascript
剖析Angular Component的源码示例
2018/03/23 Javascript
VUE实现可随意拖动的弹窗组件
2018/09/25 Javascript
浅析vue-cli3配置webpack-bundle-analyzer插件【推荐】
2019/10/23 Javascript
vue项目如何监听localStorage或sessionStorage的变化
2021/01/04 Vue.js
python批量修改文件后缀示例代码分享
2013/12/24 Python
在Python中使用zlib模块进行数据压缩的教程
2015/06/26 Python
Python使用re模块实现信息筛选的方法
2018/04/29 Python
Python(TensorFlow框架)实现手写数字识别系统的方法
2018/05/29 Python
python使用turtle库绘制树
2018/06/25 Python
对python 读取线的shp文件实例详解
2018/12/22 Python
Django 过滤器汇总及自定义过滤器使用详解
2019/07/19 Python
python3光学字符识别模块tesserocr与pytesseract的使用详解
2020/02/26 Python
简单了解Python write writelines区别
2020/02/27 Python
教师考察材料范文
2014/06/03 职场文书
公共场所禁烟标语
2014/06/25 职场文书
篮球社团活动总结
2014/06/27 职场文书
委托书格式范文
2015/01/28 职场文书
特岗教师个人总结
2015/02/10 职场文书
生产车间主任岗位职责
2015/04/08 职场文书
实习感想范文
2015/08/10 职场文书
opencv 分类白天与夜景视频的方法
2021/06/05 Python