小白学 Python(20):迭代器基础
人生苦短,我选Python
前文传送门
迭代器
迭代器可以简单的理解为 for 循环, Python 除了 for 循环为我们准备了另一种访问集合元素的方式。
特点:
- 可以记住遍历的位置的对象。
- 迭代器从集合的第一个元素开始访问,直到所有的元素访问结束。
- 迭代器只能向前不能后退。
但是判断一个对象是否有迭代器,除了看它能不能使用 for 循环外, Python 也为我们提供了更加专业的方法—— isinstance()
。
我们可以使用 isinstance()
来判断当前的对象是否可以迭代。
在使用迭代器之前,需要先将迭代器引入,因为迭代器不是 Python 的内置方法。
from collections.abc import Iterable
print(isinstance('geekdigging', Iterable))
print(isinstance([], Iterable))
print(isinstance([], Iterable))
print(isinstance({x for x in range(5)}, Iterable))
print(isinstance(123, Iterable))
打印结果如下:
True
True
True
True
False
这里有一个小点注意一下,当我们在引入 Iterable
的时候,在之前的版本引入方式都是 from collections import Iterable
,但是在 Python3.7 的时候抛了一个异常出来,如下:
D:/Development/Projects/python-learning/base-iter/Demo.py:1: DeprecationWarning: Using or importing the ABCs from 'collections' instead of from 'collections.abc' is deprecated, and in 3.8 it will stop working
from collections import Iterable
含义是这种引入方式将在 Python3.8 中停止使用,现在 Python3.8 已经发布了,我们就按照提示中的方式来引入,使用 from collections.abc import Iterable
。
迭代器中,最重要的方法一定是 next() ,从名字上就能看出来,它的作用是下一个。
我们现在来尝试迭代一个列表:
list1 = [1, 2, 3, 4]
next(list1)
执行结果:
Traceback (most recent call last):
File "D:/Development/Projects/python-learning/base-iter/Demo.py", line 12, in <module>
next(list1)
TypeError: 'list' object is not an iterator
不是说好的列表是可以迭代的么,为什么又报错了。
这位同学,先消消气,看看报错信息说的是啥,这是在提示我们列表并不是一个迭代器。
所以,可迭代对象不一定是迭代器!
我们导入 Iterator
模块,先将列表转换成迭代器,再试试看。
list1 = iter(list1)
print(type(list1))
执行结果:
<class 'list_iterator'>
从类型上来看,现在已经是一个可以迭代的列表了,我们再试试 next()
方法。
print(next(list1))
print(next(list1))
print(next(list1))
print(next(list1))
执行结果:
1
2
3
4
你看,现在就能正常打印了吧。
刚才我们将整个列表中的元素迭代完了,如果这时候再迭代一次会怎么样呢?
print(next(list1))
执行结果:
Traceback (most recent call last):
File "D:/Development/Projects/python-learning/base-iter/Demo.py", line 23, in <module>
print(next(list1))
StopIteration
看到了咯,会抛出 StopIteration
的异常。
可能看到这里,大家还是感觉迭代器没什么用,下面我再来举个例子,如果我们现在有一个 set 集合,我想要获取一个一个 set 集合中的元素,这时迭代器就派上用场了,因为在 set 集合中,是没有索引的,如果想要取到某个值,就只能使用 for 循环整个集合,但是有了迭代器以后,我们就无需循环整个集合了。
set1 = {1, 2, 3, 4, 5}
set1 = iter(set1)
print(next(set1))
print(next(set1))
print(next(set1))
print(next(set1))
print(next(set1))
结果我就不放出来了,大家应该都知道结果了。
除此之外,迭代器有一个非常巨大的优势是在迭代之前,无需准备好所有的元素,迭代器仅在迭代到某个元素的时候才开始计算这个元素,在这之前或者之后,元素是可以不存在或者被销毁的。
这个特性使得它在遍历一些体积巨大的集合的时候,具有非常大的优势。
今天的内容到这里就结束,希望各位同学可以自己动手尝试一下迭代其他集合类型,如元组,字典等。
感谢各位同学长久以来对我的支持,你们的转发和关注是我最大的动力。
示例代码
本系列的所有代码小编都会放在代码管理仓库 Github 和 Gitee 上,方便大家取用。
作者:极客挖掘机
定期发表作者的思考:技术、产品、运营、自我提升等。
本文版权归作者极客挖掘机和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
如果您觉得作者的文章对您有帮助,就来作者个人小站逛逛吧:极客挖掘机
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· DeepSeek如何颠覆传统软件测试?测试工程师会被淘汰吗?