搞清楚 Python 的迭代器、可迭代对象、生成器

本文详细介绍了Python中的迭代器协议、可迭代对象和生成器的概念。迭代器协议要求对象实现`iter()`和`next()`方法。可迭代对象是实现了`iter()`方法并返回迭代器的对象,如列表。手写迭代器类展示了如何实现这两个方法。生成器是迭代器的一种,可通过生成器函数或生成器表达式创建。迭代器具有惰性、无限元素支持和节省空间的特点。文章还提醒注意迭代器只能遍历一次,以及迭代器同时也是可迭代对象的特性。

很多伙伴对 Python 的迭代器、可迭代对象、生成器这几个概念有点搞不清楚,我来说说我的理解,希望对需要的朋友有所帮助。

1 迭代器协议

迭代器协议是核心,搞懂了这个,上面的几个概念也就很好理解了。

所谓迭代器协议,就是要求一个迭代器必须要实现如下两个方法

iterator.iter()

Returnthe iteratorobjectitself.

iterator.next()

Returnthenextitemfromthe container.

也就是说,一个对象只要支持上面两个方法,就是迭代器。 iter() 需要返回迭代器本身,而 next() 需要返回下一个元素。

2 可迭代对象

知道了迭代器的概念,那可迭代对象又是啥呢?

这个更简单,只要对象实现了 iter() 方法,并且返回的是一个迭代器,那么这个对象就是可迭代对象。

比如我们常见的列表就是可迭代对象

l = [1,3,5]>>> iter(l)复制代码

使用 iter() 会调用对应的 iter() 方法,这里返回的是一个列表迭代器,所以说列表就是一个可迭代对象。

3 手写一个迭代器

迭代器的实现有不同的方式,相信大家首先能想到的就是自定义类,我们就从这个说起。

便于说明,我们手写一个迭代器,用于生成奇数序列。

按照迭代器协议,我们实现上述的两个方法。

classOdd:def__init__(self, start=1):self.cur = startdef__iter__(self):returnselfdef__next__(self):ret_val = self.cur self.cur +=2returnret_val复制代码

终端里,我们实例化一个 Odd 类得到一个对象 odd

odd = Odd()>>> odd复制代码

使用 iter() 方法会调用类里的 iter 方法,得到它本身

iter(odd)复制代码

使用 next() 方法会调用对应的 next() 方法,得到下一个元素

next(odd)1>>> next(odd)3>>> next(odd)5复制代码

其实,odd 对象就是一个迭代器了。

我们可以用 for 来遍历它

odd = Odd()forvinodd:print(v)复制代码

细心的伙伴可能会发现,这个其实会无限的打印下去,那怎么解决呢?

我们拿一个列表做做实验,先得到它的迭代器对象

l = [1,3,5]>>> li = iter(l)>>> li复制代码

然后手动获取下一个元素,直到没有下一个元素为止,看下会发生什么

next(li)1>>> next(li)3>>> next(li)5>>> next(li)Traceback (most recent call last): File"", line1,inStopIteration复制代码

原来列表迭代器会在没有下一个元素的时候抛出 StopIteration 异常,估计 for 语句就是根据这个异常来确定是否结束。

我们修改一下原来的代码,能生成指定范围内的奇数

classOdd:def__init__(self, start=1, end=10):self.cur = start self.end = enddef__iter__(self):returnselfdef__next__(self):ifself.cur > self.end:raiseStopIteration ret_val = self.cur self.cur +=2returnret_val复制代码

我们使用 for 试一下

odd = Odd(1,10)>>> forvinodd:… print(v)…13579复制代码

果然,和预期一致。

我们用 while 循环模拟 for 的执行过程

目标代码

forviniterable:print(v)复制代码

翻译后的代码

iterator = iter(iterable)whileTrue:try:v = next(iterator)print(v)exceptStopIteration:break复制代码

事实上 Python 的 for 语句原理也就是这样,可以将 for 理解为一个语法糖。

4 创建迭代器的其它方式

生成器其实也是迭代器,所以可以使用生成器的创建方式创建迭代器。

4.1 生成器函数

和普通函数的 return 返回不同,生成器函数使用 yield。

defodd_func(start=1, end=10):… forvalinrange(start, end +1):… ifval %2==1:… yieldval…>>> of = odd_func(1,5)>>> of>>> iter(of)>>> next(of)1>>> next(of)3>>> next(of)5>>> next(of)Traceback (most recent call last): File"", line1,inStopIteration复制代码

4.2 生成器表达式

g = (v for v in range(1,5+1) if v %2==1)>>> g>>> iter(g)>>> next(g)1>>> next(g)3>>> next(g)5>>> next(g)Traceback (most recent call last): File"", line1,inStopIteration复制代码

4.3 怎么选择

到现在为止,我们知道了创建迭代器的 3 种方式,那么该如何选择?

不用说也知道,最简单的就是生成器表达式,如果表达式能满足需求,那么就是它;如果需要添加比较复杂的逻辑就选生成器函数;如果前两者没法满足需求,那就自定义类实现吧。总之,选择最简单的方式就行。

5 迭代器的特点

5.1 惰性

迭代器并不是把所有的元素提前计算出来,而是在需要的时候才计算返回。

5.2 支持无限个元素

比如上面我们建立的第一个 Odd 类,它的实例 odd 表示大于 start 的所有奇数,而列表等容器没法容纳无限个元素的。

5.3 省空间

比如存 10000 个元素

from sys import getsizeof>>> a = [1] *10000>>> getsizeof(a)80064复制代码

列表占用 80K 左右。

而迭代器呢?

from itertools import repeat>>> b = repeat(1, times=10000)>>> getsizeof(b)56复制代码

只占用了 56 个字节。

也正因为迭代器惰性的特点,才有了这个优势。

6 一些需要注意的细节

6.1 迭代器同时也是可迭代对象

因为迭代器的 iter() 方法返回了它自身,而正好它本身就是个迭代器,所以说迭代器也是可迭代对象。

6.2 迭代器遍历完一次就不能从头开始了

看一个奇怪的例子

l = [1,3,5]>>> li = iter(l)>>> li>>>3in liTrue>>>3in liFalse复制代码

因为 li 是列表迭代器,第一次查找 3 的时候,找到了,所以返回 True,但是由于第一次迭代,已经跳过了 3 那个元素,第二次就找不到了,所以会出现 False。

因此,记得迭代器是「一次性」的。

当然,列表是可迭代对象,不管查找几次都是正常的。(不好理解的话,想想上面 for 语句的执行原理,每次都会从可迭代对象那通过 iter() 方法取到新的迭代器)

3inlTrue>>> 3inlTrue复制代码

7 小节

iter()

前面 3 小项是重点,这 3 点理解了,其它的也都能领会。搞清楚标题的那几个名词的概念的自然也没有问题。

代码转载自:https://pan.quark.cn/s/8ce4326d996e 对于在 CentOS 7 系统中修改网卡配置文件后无法使设置生效的情况,经过实践验证,可以通过使用 nmcli 命令来进行调整。完成修改之后,需要重新启动虚拟机以使更改生效,这样操作流程即告完成。如果设置仍然无法生效,则表明虚拟机在启动过程中所获取的 IP 地址配置并非针对 eth0,此时可以对其它网卡的配置文件进行修改或将其移除。在 CentOS 7 系统中,网络配置的管理机制与早期版本存在差异,主要体现为采用了 Network Manager 服务来负责网络接口的管理。在某些情形下,尽管修改了 `/etc/sysconfig/network-scripts` 目录下的 `ifcfg-eth0` 文件,但网络配置却未能即时生效。此类问题的发生通常源于 CentOS 7 采用了不同于以往的配置读取方法。接下来将具体阐述如何借助 nmcli 命令来处理这一挑战。 以 root 用户身份登录系统并打开终端界面。nmcli 是 Network Manager 提供的命令行界面工具,它支持在命令行环境下执行网络连接的建立、编辑、查询及管理任务。针对修改 eth0 网卡配置的需求,可以遵循以下步骤进行操作: 1. 导航至 `/etc/sysconfig/network-scripts` 目录: ``` cd /etc/sysconfig/network-scripts ``` 2. 检查该目录内是否存在 `ifcfg-eth0.bak` 文件,该备份文件可能是先前调整配置时遗留下来的,若存在可能造成冲突。若发现该文件,可以选择将其删除: ``` [root@localhost netw...
代码转载自:https://pan.quark.cn/s/46fd08fb879c 网管教程 从入门到精通软件篇 ★一。★详尽的xp修复控制台指令及其应用!!! 放入xp(2000)的光盘,安装时选择R,执行修复! Windows XP(涵盖 Windows 2000)的控制台指令是在系统遭遇某些意外状况时的一种极具效用的诊断、检测以及恢复系统功能的工具。笔者确实一直期望能够将这方面的指令进行归纳,此次由老范辛苦整理了这份极具价值的秘籍。 Bootcfg bootcfg 命令用于启动配置与故障恢复(对大多数计算机而言,即 boot.ini 文件)。 带有特定参数的 bootcfg 命令仅在运用故障恢复控制台时方可使用。能够在命令行界面下运用带有不同参数的 bootcfg 命令。 用法: bootcfg /default 设定默认引导选项。 bootcfg /add 向引导清单中增添 Windows 安装。 bootcfg /rebuild 重复整个 Windows 安装流程并让用户选择需添加的项目。 注意:运用 bootcfg /rebuild 之前,应先借助 bootcfg /copy 命令备份 boot.ini 文件。 bootcfg /scan 探查用于 Windows 安装的全部磁盘并展示结果。 注意:这些结果被静态存储,并用于当前会话。若在当前会话期间磁盘配置发生变动,为获取更新的探查结果,必须先重启计算机,然后再次探查磁盘。 bootcfg /list 列示引导清单中已有的项目。 bootcfg /disableredirect 在启动引导程序中禁用重定向。 bootcfg /redirect [ PortBaudRrate] |[ useBio...
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值