一、迭代器
迭代器是访问集合元素的一种方式。
迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束。
可以被next()函数调用并不断返回下一个值的对象称为迭代器:Iterator。
迭代器的特点:
- 访问者不需要关心迭代器内部的结构,仅需通过next()方法不断去取下一个内容
- 不能 随机访问集合中的某个值 ,只能从头到尾依次访问
- 访问到一半时不能往回退
- 便于循环比较大的数据集合,节省内存
小虾刚刚说了一些迭代器的特点,那么如果需要迭代器的话,肯定会涉及到迭代对象,那么接下来让我们看一下迭代对象都是有哪些吧!
可迭代对象分为两大类:
一种是实际保存的序列,即列表、元组,字符串;
另一种就是我们上一节总提到的“不一次性产生所有结果列表,而是可以在for循环中按需一次产生一个结果的对象”。如:range函数返回值、zip函数返回值、enumerate函数返回值等等,与实际序列相对应,这个叫做按需产生对象的虚拟序列。
接下来小虾就说一下迭代器与可迭代对象的关系:
可迭代对象支持内置函数iter,通过对可迭代对象调用iter函数,会返回一个迭代器,而“迭代器”支持内置函数next,通过不断对其调用next方法,会依次前进到序列中的下一个元素并将其返回,最后到达一系列结果的末尾时,会引发StopIteration异常。补充说明一点:对迭代器调用iter方法,则会返回迭代器自身。
S = [11,2,3]
L = iter(S)
print(iter(S))
print(L is S)
print(L is iter(I))
运行结果为:
<list_iterator object at 0x00000000021E0978>
False
True
从上面例子中我们看到:通过iter函数将迭代对象转化成迭代器,而对迭代器调用iter函数,依然返回迭代器。
具体的迭代过程就是下面的一句话:当任何可迭代对象传入到for循环或其他迭代工具中进行遍历时,迭代工具都是先通过iter函数获得与可迭代对象对应的迭代器,然后再对迭代器调用next函数,不断的依次获取元素,并在捕捉到StopIteration异常时确定完成迭代,这就是完整的迭代过程。
下面在举几个小例子,可以学习一下:
1.迭代enumerate函数返回对象(enumerate方法返回的也是可迭代对象,他的迭代器就是他自身)
E = enumerate('spam')
print(E)
print(iter(E))
运行结果:
<enumerate object at 0x00000000021F2558>
<enumerate object at 0x00000000021F2558>
如果我们想看到enumerate函数中的值,那么我们必须把可迭代对象包裹在一个list调用中,从而才能一次性看到它们所有的值
print(list(enumerate('spam')))
运行结果为:
[(0, 's'), (1, 'p'), (2, 'a'), (3, 'm')]
2.迭代字典对象(字典也是一种可迭代对象,但是他的迭代器却比较特殊)
D = {"a":1,"b":2,"c":3}
I = iter(D)
print(next(I))
print(next(I))
print(next(I))
print(next(I))
运行结果为:
a
b
c
Traceback(most recent call last):
File"D:/homework/homework.py", line 6, in <module>
print(next(I))
StopIteration
因此不难看出,字典也是一个可迭代对象,字典有一个迭代器,在迭代环境中,会每次自动地返回一个键。
总结:
- 凡是可作用于
for
循环的对象都是Iterable
类型; - 凡是可作用于
next()
函数的对象都是Iterator
类型,它们表示一个惰性计算的序列; - 集合数据类型如
list
、dict
、str
等是Iterable
但不是Iterator
,不过可以通过iter()
函数获得一个Iterator
对象。
不后悔过去,不幻想未来,把握好现在,就是对自己最好的交代!!!!!!!!!!!!