迭代 生成
for循环遍历的原理
for循环遍历的原理就是迭代,in后面必须是可迭代对象
为什么要有迭代器
对于序列类型:字符串、列表、元组,我们可以使用索引的方式迭代取出其包含的元素。但对于字典、集合、文件等类型是没有索引的,若还想取出其内部包含的元素,则必须找出一种不依赖于索引的迭代方式,这就是迭代器
1.可迭代对象
有__iter__
方法的对象,都是可迭代对象,有以下6种
可迭代的对象:Python内置str、list、tuple、dict、set、file都是可迭代对象
"zx".__iter__()
["zx"].__iter__()
{"zx":"wl"}.__iter__()
("zx",).__iter__()
{"z","x"}.__iter__()
with open("prize.txt","r") as file:
file.__iter__()
2.迭代器对象
1.可迭代的对象执行__iter__
方法得到的返回值是迭代器对象。2.迭代器对象指的是即内置有__iter__
又内置有__next__
方法的对象
list=[1,2,3,4,5,6]
zx=list.__iter__()
while True:
try:
print(zx.__next__())
except:
break
文件类型是迭代器对象
open('a.txt').__iter__()
open('a.txt').__next__()
迭代器对象一定是可迭代对象,而可迭代对象不一定是迭代器对象
生成器
生成器就是一种迭代器。迭代器有一个特点就是可以被next()函数调用并不断返回下一个值的对象,并且只能迭代它们一次。原因很简单,因为它们不是全部存在内存里,它们只在要调用的时候在内存里生成。
生成器就是迭代器(函数返回的结果就是生成器,而且它同时有__iter__
和__next__
)
def zx():
yield 1
yield 2
x=zx()
x.__next__()
x.__iter__()
实验室
每次__next__
之后,会执行到相应的yield不会执行下面的内容。
为啥?这个实验执行了呢,因为用了for,他不知道最后执行到哪里,他就会一直__next__
,直到最后抛出异常,然而for in里面自带捕获异常的内容,所有没有打印异常信息。
def func():
yield [1,1,23] # yield会使函数func()变成生成器对象,因此他就具有__iter__方法
print(789) # yield会停止函数,当运行下一次next才会继续运行下面的代码
yield 101112 # 一个yield对应一个next
print(131415)
g = func()
for i in g:
print(i)
[1, 1, 23]
789
101112
131415
关于迭代器和和生成器的区别
生成器是一种特殊的迭代器,生成器实现了迭代器协议--iter--,--next--
生成器是可以改变迭代的值的,然而迭代器随意改值会有问题
关于迭代器和list的区别
list直接把所有数据加载到内存
而迭代器是一个一个取值,在需要下一个值的时候才回去计算取出这个值到内存(可以把迭代器想象成一个生成器的代码,一次next,运行下面的代码,然后返回值)