迭代
迭代是访问容器元素的一种方式
迭代器是一个可以记住遍历的位置的对象
迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束
可迭代对象
可以直接作用于for循环的对象(如何判断是否可以迭代?)
一类是集合数据类型,如list, tuple,dict, set,str等
一类是generator,包括生成器和带yield的generator function
迭代器
可以被next()函数调用并不断返回下一个值的对象称为迭代器:Iterator。
生成器都是Iterator对象,但list、dict、str虽然是Iterable,却不是Iterator。
把list、dict、str等Iterable变成Iterator可以使用iter()函数
你可能会问,为什么list、dict、str等数据类型不是Iterator?
这是因为Python的Iterator对象表示的是一个数据流,Iterator对象可以被next()函数调用并不断返回下一个数据,直到没有数据时抛出StopIteration错误。可以把这个数据流看做是一个有序序列,但我们却不能提前知道序列的长度,只能不断通过next()函数实现按需计算下一个数据,所以Iterator的计算是惰性的,只有在需要返回下一个数据时它才会计算。
Iterator甚至可以表示一个无限大的数据流,例如全体自然数。而使用list是永远不可能存储全体自然数的
迭代器和生成器的区别 ?
将字符串转成迭代器
可以调用next方法的对象
# 将字符串转成迭代器(可以调用next方法的对象)
str = 'hello'
print(type(str))
str_iter = iter(str)
print(str_iter)
print(type(str_iter))
# 运行结果
<class 'str'>
<str_iterator object at 0x7f36235d6dd8>
<class 'str_iterator'>
调用next方法遍历
# 调用next方法遍历
loop0 = str_iter.__next__()
print(loop0)
loop1 = str_iter.__next__()
print(loop1)
loop0 = str_iter.__next__()
print(loop0)
loop1 = str_iter.__next__()
print(loop1)
loop1 = str_iter.__next__()
print(loop1)
loop1 = str_iter.__next__() # 执行到词语会出错,因为字符串已经遍历完了
print(loop1)
# 运行结果
h
e
l
l
o
Traceback (most recent call last):
File "/home/kiosk/PycharmProjects/2019PythonProject/week06/day01/iterator.py", line 18, in <module>
loop1 = str_iter.__next__()
StopIteration
迭代器如何实现for循环?
# 字符串遍历
while True:
try:
loop = str_iter.__next__()
print(loop)
except:
print("遍历结束")
break
# 运行结果
h
e
l
l
o
遍历结束
为什么第二次遍历字符串没有内容呢?
因为next方法从上一次停止的地方继续执行,只能向后算,可以重新生成一个迭代器来解决
while True:
try:
loop = str_iter.__next__()
print(loop)
except:
print("遍历结束")
break
while True:
try:
loop = str_iter.__next__()
print(loop)
except:
print("字符串遍历结束")
break
str_iter = iter(str)
while True:
try:
loop = str_iter.__next__()
print(loop)
except:
print("字符串遍历结束")
break
# 运行结果
h
e
l
l
o
遍历结束
字符串遍历结束
h
e
l
l
o
字符串遍历结束
如何判断一个对象是不是可迭代对象?
from collections.abc import Iterable
print(type('hello') == str)
# isinstance返回对象是类的实例还是其子类的实例
print(isinstance('hello', str))
print(isinstance(1, Iterable))
print(isinstance('hello', Iterable))
#运行结果
True
True
False
True
文件对象是不是可迭代对象?
# 代码
with open('/home/kiosk/PycharmProjects/2019PythonProject/week05/day01/copy_file.txt', 'r') as f:
print(isinstance(f, Iterable))
# 读取大文件的方式: > 4G,一行一行读取
# enumerate: 枚举, 返回元素索引和元素值
for number, line in enumerate(f):
print("第%d行:" %(number+1), line, end='')
# 运行结果
True
第1行: abcdefg123456
第2行: hijklmn111111