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后面跟了一大串字符 data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAYAAA...

    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