当前位置: 首页 > news >正文

Python 迭代器

迭代器协议

对象必须提供一个 next() 方法,执行该方法要么迭代下一项,要么就引起一个 StopIteration异常以终止迭代(只能往后不能往前)—— 迭代器协议

协议是一种约定,可迭代对象实现了迭代器协议(for、sum、min、max 等使用迭代器协议访问对象)

可迭代对象(iterable)

实现了迭代器协议的对象,就是可迭代对象。如何实现迭代器协议:对象内部定义 __iter__() 方法。

可作用于 for 循环的对象统称为可迭代对象,能作用于 for 循环的数据类型有以下几种:

  • 集合数据类型:liststrtupledicttuple
  • 生成器 generator:包括生成器函数和带 yield 关键字的生成器对象

可以用 isinstance() 判断一个对象是否是可迭代对象:

from collections import Iterableisinstance([], Iterable)			# True
isinstance('123', Iterable)			# True
isinstance(123, Iterable)			#  False

可迭代对象是可迭代的(即能被 for 循环),但不是迭代器,因为 next() 函数无法调用它们:

next([1, 2, 3])
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
<ipython-input-38-4a6488f07b8e> in <module>()
----> 1 next([1, 2, 3])TypeError: 'list' object is not an iterator		# list 对象不是一个迭代器

但是可以用 iter() 函数将其转换为迭代器:

l = [1, 2, 3]
ret = iter(l)		# 转换为迭代器
next(ret)			# 调用 next() 函数------------------------
1

迭代器(iterator)

可以被next()函数调用并不断返回下一个值的对象称为迭代器Iterator

迭代器对象从集合第一个元素开始访问,直到所有元素被访问完才算结束,只能往前不能往后。

使用isinstance()判断一个对象是否是Iterator对象:

from collections import Iterator
isinstance([], Iterator)        					# list 不是迭代器对象,False
isinstance(iter([1, 2, 3]), Iterator)   			# 用 iter() 将 list 转换为迭代器,True
isinstance((x for x in range(10)), Iterator)   		# 生成器是迭代器对象

迭代器惰性机制

迭代器表示的是一个数据流,可以调用 next() 函数不断返回下一个数据,直至没有数据抛出 StopIteration。可以说它是一个有序序列,但我们不知道序列长度,只能通过 next() 函数实现按需计算下一个数据,所有它是惰性的,只有在需要时才计算返回下一个数据。


为什么要有迭代器

对于有序序列类型,使用索引以及 while 循环也难遍历所有元素,但是对于无序序列类型(字典、集合)就会束手无策。


for 循环机制

for 循环是基于迭代器协议提供了一个统一的可遍历所有对象的方法,调用对象的 __iter__() 方法将其转换为一个迭代器,再调用 __next__() 不断取出序列中元素。相比较迭代器而已,for 循环还完成了 StopIteration 的异常工作。

这样一来,只要是序列,都可以通过 for 循环遍历,不管是有序还是无序序列类型。


优缺点

优点:

  • 提供一种不依赖于索引的迭代方式
  • 惰性计算,节省内存(每次取值,才会加载)

缺点:

  • 无法获取长度,next() 完毕才知道
  • 一次性,只能往前,不能往后

自定义类为迭代器

要想将自定义的类实现为一个迭代器,需要在类中实现 __iter__()__next__()方法:

  • __iter__() 方法:返回迭代器对象本身
  • __next__() 方法:返回容器中下一个元素,当超出边界时触发 StopIteration 异常终止迭代器
class Foo(object):def __init__(self,step):self.step = stepdef __next__(self):if self.step == 0:raise StopIterationself.step -= 1return self.stepdef __iter__(self):return self
f = Foo(6)
for i in f:print(i)
5
4
3
2
1
0

总结

  • 可以 for 循环的对象都是可迭代类型
  • 可以作用于 next() 函数的对象都是迭代器类型,它表示一个惰性计算的序列
  • listdictstr 等都是可迭代的,但不是迭代器类型,因为没有 next() 方法,可以用 iter() 函数将其转换为迭代器
  • Python 的 for 循环机制其实质是不断调用 next() 函数实现的,但是它超出边界不会触发 StopIteration
  • for 循环一次将所有数取出,迭代器一次只取一个
http://www.lryc.cn/news/12982.html

相关文章:

  • Python语言零基础入门教程(二十七)
  • Redis基础操作以及数据类型
  • 自抗扰控制ADRC之反馈控制律(NLSEF)
  • “生成音乐“ 【循环神经网络】
  • 能否手写vue3响应式原理-面试进阶
  • 前端工程师leetcode算法面试必备-简单的二叉树
  • 【什么程度叫熟悉linux系统】
  • 编译安装MySQL
  • Kubernetes一 Kubernetes之入门
  • SQLServer2000 断电后数据库suspect“置疑”处理
  • 多模态机器学习入门Tutorial on MultiModal Machine Learning——第一堂课个人学习内容
  • Java ~ Collection/Executor ~ LinkedBlockingDeque【总结】
  • .NET7的AOT的使用
  • 分布式缓存的问题
  • golang入门笔记——内存管理和编译器优化
  • GEE学习笔记 七十:【GEE之Python版教程四】Python基础编程二
  • 股票投资新出发之知识体系构建导论
  • 蓝桥杯算法训练合集 十六 1.首字母变大写2.盾神计科导作业3.Cinema4.接水问题
  • 密码的世界
  • 如何用一句话感动测试工程师?产品和技术都这么说!
  • 3|物联网控制|计算机控制-刘川来胡乃平版|第2章:计算机控制系统中的检测设备和执行机构-2.1传感器和变送器|课堂笔记|ppt
  • MySQL中使用索引优化
  • Linux C/C++ 多线程TCP/UDP服务器 (监控系统状态)
  • 【JavaScript】JavaScript基本使用方法
  • Python数据容器、list列表、tuple元组、str字符串、数据容器(序列)切片、set集合、dict字典、字符串大小比较
  • Python urllib
  • Centos7安装Python3
  • [U3D ShaderGraph] 全面学习ShaderGraph节点 | 第四课 | Input/Lighting
  • SpringBoot升级到3.0
  • JavaWeb8-线程安全问题