Python 迭代器

2023-05-16

一、迭代器

迭代是访问集合元素的一种方式。迭代器是一个可以记住遍历的位置的对象。迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束。迭代器只能往前不会后退。

1.1 判断一个对象是否可迭代

可以使用 isinstance() 判断一个对象是否是 Iterable 对象:

In [50]: from collections import Iterable

In [51]: isinstance([], Iterable)
Out[51]: True

In [52]: isinstance({}, Iterable)
Out[52]: True

In [53]: isinstance('abc', Iterable)
Out[53]: True

In [54]: isinstance(mylist, Iterable)
Out[54]: False

In [55]: isinstance(100, Iterable)
Out[55]: False  

1.2 可迭代对象的本质

我们分析对可迭代对象进行迭代使用的过程,发现每迭代一次(即在for...in...中每循环一次)都会返回对象中的下一条数据,一直向后读取数据直到迭代了所有数据后结束。那么,在这个过程中就应该有一个“人”去记录每次访问到了第几条数据,以便每次迭代都可以返回下一条数据。我们把这个能帮助我们进行数据迭代的“人”称为迭代器(Iterator)。
 
可迭代对象的本质就是可以向我们提供一个这样的中间“人”即迭代器帮助我们对其进行迭代遍历使用。
 
可迭代对象通过__iter__方法向我们提供一个迭代器, 我们在迭代一个可迭代对象的时候,实际上就是先获取该对象提供的一个迭代器,然后通过这个迭代器来依次获取对象中的每一个数据.
 
那么也就是说,一个具备了__iter__方法的对象,就是一个可迭代对象

 1 from collections import Iterable
 2 
 3 
 4 class MyList(object):
 5 
 6     def __init__(self):
 7         self.container = []
 8 
 9     def add(self, item):
10         self.container.append(item)
11 
12     def __iter__(self):
13         """返回一个迭代器"""
14         pass
15 
16 
17 mylist = MyList()
18 print(isinstance(mylist, Iterable))     # True  

这回测试发现添加了__iter__方法的mylist对象已经是一个可迭代对象了

1.3 iter()函数和next()函数

list、tuple等都是可迭代对象,我们可以通过iter()函数获取这些可迭代对象的迭代器。然后我们可以对获取到的迭代器不断使用next()函数来获取下一条数据。iter()函数实际上就是调用了可迭代对象的__iter__方法。

>>> li = [11, 22, 33, 44, 55]
>>> li_iter = iter(li)
>>> next(li_iter)
11
>>> next(li_iter)
22
>>> next(li_iter)
33
>>> next(li_iter)
44
>>> next(li_iter)
55
>>> next(li_iter)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
StopIteration
>>>  

注意,当我们已经迭代完最后一个数据之后,再次调用next()函数会抛出StopIteration的异常,来告诉我们所有数据都已迭代完成,不用再执行next()函数了。

1.4 如何判断一个对象是否是迭代器

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


In [56]: from collections import Iterator

In [57]: isinstance([], Iterator)
Out[57]: False

In [58]: isinstance(iter([]), Iterator)
Out[58]: True

In [59]: isinstance(iter("abc"), Iterator)
Out[59]: True  

1.5 迭代器Iterator

通过上面的分析,我们已经知道,迭代器是用来帮助我们记录每次迭代访问到的位置,当我们对迭代器使用next()函数的时候,迭代器会向我们返回它所记录位置的下一个位置的数据。实际上,在使用next()函数的时候,调用的就是迭代器对象的__next__方法(Python3中是对象的__next__方法,Python2中是对象的next()方法)。所以,我们要想构造一个迭代器,就要实现它的__next__方法。但这还不够,python要求迭代器本身也是可迭代的,所以我们还要为迭代器实现__iter__方法,而__iter__方法要返回一个迭代器,迭代器自身正是一个迭代器,所以迭代器的__iter__方法返回自身即可。
 
一个实现了__iter__方法和__next__方法的对象,就是迭代器。

 1 # 实现迭代器(两个类的写法)
 2 class MyList(object):
 3     """
 4     自定义的一个可迭代对象
 5     """
 6 
 7     def __init__(self):
 8         self.items = []
 9 
10     def add(self, val):
11         self.items.append(val)
12 
13     def __iter__(self):
14         myiterator = MyIterator(self)
15 
16         return myiterator
17 
18 
19 class MyIterator(object):
20     """
21         自定义的供上面可迭代对象使用的一个迭代器
22     """
23     def __init__(self, mylist):
24         self.mylist = mylist
25         # current用来记录当前访问到的位置
26         self.current = 0
27 
28     def __next__(self):
29         if self.current < len(self.mylist.items):
30             item = self.mylist.items[self.current]
31             self.current += 1
32             return item
33         else:
34             raise StopIteration
35 
36     def __iter__(self):
37         return self
38 
39 
40 if __name__ == '__main__':
41     mylist = MyList()
42     mylist.add(1)
43     mylist.add(2)
44     mylist.add(3)
45     mylist.add(4)
46     mylist.add(5)
47     for num in mylist:
48         print(num)  

 1 实现迭代器(写在一个类) 
 2 class MyList(object):
 3 
 4 
 5     def __init__(self):
 6         self.container = []
 7         self.current = 0
 8 
 9 
10     def add(self, item):
11         self.container.append(item)
12 
13 
14     def __iter__(self):
15 
16 
17         return self
18 
19 
20     def __next__(self):
21         if self.current < len(self.container):
22             item = self.container[self.current]
23             self.current += 1
24             return item
25         else:
26             raise StopIteration
27 
28 
29 
30 
31 my_list = MyList()
32 my_list.add(1)
33 my_list.add(2)
34 my_list.add(3)
35 my_list.add(4)
36 my_list.add(5)
37 
38 
39 for num in my_list:
40     print(num)  

1.6 for...in...循环的本质

for item in Iterable 循环的本质就是先通过iter()函数获取可迭代对象Iterable的迭代器,然后对获取到的迭代器不断调用next()方法来获取下一个值并将其赋值给item,当遇到StopIteration的异常后循环结束。
 
迭代器的特点:
  • 节省内存
  • 惰性机制
  • 不能反复,只能向下

1.7 迭代器的应用场景

斐波那契数列


 1 class FibIterator(object):
 2     """
 3         斐波那契数列
 4     """
 5 
 6     def __init__(self, n):
 7         """
 8         :param n:int, 指明生成数列的前n个数
 9         """
10         self.n = n
11         # current用来保存当前生成到数列中的第几个数了
12         self.current = 0
13         # num1用来保存前前一个数,初始值为数列中的第一个数0
14         self.num1 = 0
15         # num2用来保存前一个数,初始值为数列中的第二个数1
16         self.num2 = 1
17 
18     def __next__(self):
19         """
20         被next()函数调用来获取下一个数
21         :return:
22         """
23 
24         if self.current < self.n:
25             num = self.num1
26             self.num1, self.num2 = self.num2, self.num1+self.num2
27             self.current += 1
28             return num
29         else:
30             raise StopIteration
31 
32     def __iter__(self):
33         """
34         "迭代器的__iter__返回自身即可
35         :return:
36         """
37         return self
38 
39 
40 if __name__ == '__main__':
41     fib = FibIterator(10)
42     for num in fib:
43         print(num, end=" ")  

除了for循环能接收可迭代对象,list、tuple等也能接收。


1 除了for循环能接收可迭代对象,list、tuple等也能接收。
2 
3 li = list(FibIterator(15))
4 print(li)
5 tp = tuple(FibIterator(6))
6 print(tp)  

 

转载于:https://www.cnblogs.com/fengyuhao/p/11541630.html

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Python 迭代器 的相关文章

随机推荐

  • c# 线程的等待(堵塞)

    这里我要强调一个概念 xff0c 多线程是多线程 xff0c 异步编程是异步编程 这两个是有区别的概念 xff1b 我可以说多线程天生就要异步的特点 xff1b 但你不能说多线程成就等同于我们的异步编程 xff1b 根不能说异步编程就是我们
  • char **argv 与char *argv[]

    1 char argv 分析 xff1a argv是一个指针变量 xff0c argv的指向 xff08 argv xff09 是char xff0c 也就是argv指向的也是一个指针 xff1b argv的指向 xff08 argv xf
  • 网页上有些图片的src或css背景图片的url后面跟了一大串字符 ...

    span style color 000000 lt DOCTYPE html gt lt html dir 61 34 ltr 34 lang 61 34 en US 34 gt lt BEGIN head gt lt head gt l
  • E-mail Etiquette

    E Mail Etiquette Introduction This document is intended to offer guidance to users of electronic mail e mail systems Thi
  • linux文件系统镜像 结构,ROMFS文件系统分析[三]ROMFS映像文件结构

    上一节 On disk inode struct romfs inode be32 next low 4 bits see ROMFH be32 spec be32 size be32 checksum char name 0 初看 xff
  • centos6.7搭建局域网ntp服务器

    修改 etc ntp conf文件 restrict xxx nomodify notrap nopeer noquery xxx 此处配置本地IP地址 restrict 127 0 0 1 restrict xxx mask 255 25
  • 单片机的串口通信---查询和中断

    一 单片机与外设的通信有三种方法 xff1a 查询 xff08 除了while函数外 xff0c 还可以开启内核或者应用层线程不断轮询 xff09 xff0c 中断和DMA xff0c 这三种方式中 xff0c 查询的话你可以理解为循环检测
  • h5-面试题

    干货 各种常见布局实现 43 知名网站实例分析 前端面试考点多 xff1f 看这些文章就够了 xff08 2019年6月更新版 xff09 前端面试 xff1a 这50个经典前端面试题面试者必看 xff01 Vue面试中 xff0c 经常会
  • 导航hover延迟

    function var hoverTimer outTimer 34 nav li 34 hover function var this 61 this clearTimeout outTimer hoverTimer 61 window
  • 用ctrl+鼠标滚动调节字体大小

    如此设置之后 xff0c 按住ctrl 43 鼠标滚动 xff0c 可以放大和变小代码的字号 转载于 https www cnblogs com dengyg200891 p 6063535 html
  • CentOS7 查看操作系统版本信息

    CentOS 查看操作系统版本信息 1 使用cat proc version uname 查看内核版本 root 64 CentOS7 cat proc version Linux version 3 10 0 957 el7 x86 64
  • wincc7.4安装授权 全(文件分享)

    链接 xff1a https pan baidu com s 1YJwbvetH1gFElu468TWuAQ 提取码 xff1a 6ood 转载于 https www cnblogs com hefengweiliang p 1113633
  • Unable to resolve service for type 'Microsoft.Extensions.Logging.ILogger' while attempting to activa...

    Unable to resolve service for type 39 Microsoft Extensions Logging ILogger 39 while attempting to activate 39 xxxxx Cont
  • python的两种运行方式

    python有两种运行方式 xff0c 第一种是交互式 xff0c 另一种是脚本式 xff0c 这里可能郁闷了 xff0c 啥叫脚本 xff1f xff1f 脚本可能会想到不重要的 xff0c 边角料啥的 xff0c 我们可以认为是一个大型
  • 机器人ROS系统学习随笔->2 RPLIDAR激光雷达使用

    一 驱动的安装 SLAMTEC官网下载激光雷达资料 xff1a http www slamtec com 本人用的是RPLIDAR A1的激光雷达 在下载界面下载相应的资料 xff0c sdk与固件及其他 二 安装 1 建立工作空间并编译
  • 蓝牙飞控数传套装(适合APM/Pixhaw/Pixhack/Pixhawk2飞控)

    SSC FK BL900是迅瞻电子初创的一款蓝牙 43 射频模块一体的飞控数传套装电台 xff0c 它内嵌入Digi的900HP模块和蓝牙模块 xff0c 对外提供XT60的电池供电接口 xff0c 支持5 28V的宽电压工作 xff0c
  • VC++ 编译过程

    一 前言 一开始编译C 43 43 代码的时候可能会对编译的错误觉得很难理解 xff0c 搞不清楚究竟是哪里错了 了解编译过程 xff0c 能够更好的处理编译错误 二 名词解释 编译单元 xff1a 当一个c或cpp文件在编译时 xff0c
  • PLC的ST编程方式--文本编程,简洁啊

    一 ST语言介绍 发现网上PLC的ST编程资料极少 不过 xff0c 道理也很简单 xff0c 因为做PLC的基本都是电气出身 xff0c 梯形图类似于继电器逻辑 xff0c 比较接近他们的习惯 ST属于文本编程 xff0c 符合程序员的习
  • stm32气压传感器 带探头的_基于STM32的真空度的测量装置的设计

    阮敬华 43 张忠伟 43 徐沛 43 李雪莲 43 阚茹男 摘 要 xff1a 采用USART串口通信 SPI通信作为核心技术 xff0c 通过气压和温湿度传感器对环境的真空值 温湿度参数进行数据采集 xff0c 将采集到的数据通过SD卡
  • Python 迭代器

    一 迭代器 迭代是访问集合元素的一种方式 迭代器是一个可以记住遍历的位置的对象 迭代器对象从集合的第一个元素开始访问 xff0c 直到所有的元素被访问完结束 迭代器只能往前不会后退 1 1 判断一个对象是否可迭代 可以使用 isinstan