1.迭代器协议是指:对象必须提供一个next方法,执行该方法要么返回迭代中的下一项,要么就引发一个StopIteration异常,以终止迭代 (只能日后走不能往前退)python
2.可迭代对象:实现了迭代器协议的对象(如何实现:对象内部定义一个__iter__()方法)算法
3.协议是一种约定,可迭代对象实现了迭代器协议,python的内部工具(如for循环,sum,min,max函数等)使用迭代器协议访问对象。app
for循环的本质:循环全部对象,全都是使用迭代器协议。ide
正本清源:函数
不少人会想,for循环的本质就是遵循迭代器协议去访问对象,那么for循环的对象确定都是迭代器了啊,没错,那既然这样,for循环能够遍历(字符串,列表,元组,字典,集合,文件对象),那这些类型的数据确定都是可迭代对象啊?可是,我他妈的为何定义一个列表l=[1,2,3,4]没有l.next()方法,打脸么。工具
(字符串,列表,元组,字典,集合,文件对象)这些都不是可迭代对象,只不过在for循环式,调用了他们内部的__iter__方法,把他们变成了可迭代对象post
而后for循环调用可迭代对象的__next__方法去取值,并且for循环会捕捉StopIteration异常,以终止迭代大数据
l=['a','b','c'] #一:下标访问方式 print(l[0]) print(l[1]) print(l[2]) # print(l[3])#超出边界报错:IndexError #二:遵循迭代器协议访问方式 diedai_l=l.__iter__() print(diedai_l.__next__()) print(diedai_l.__next__()) print(diedai_l.__next__()) # print(diedai_l.__next__())#超出边界报错:StopIteration #三:for循环访问方式 #for循环l本质就是遵循迭代器协议的访问方式,先调用diedai_l=l.__iter__()方法,或者直接diedai_l=iter(l),而后依次执行diedai_l.next(),直到for循环捕捉到StopIteration终止循环 #for循环全部对象的本质都是同样的原理 for i in l:#diedai_l=l.__iter__() print(i) #i=diedai_l.next() #四:用while去模拟for循环作的事情 diedai_l=l.__iter__() while True: try: print(diedai_l.__next__()) except StopIteration: print('迭代完毕了,循环终止了') break
Python的Iterator
对象表示的是一个数据流,Iterator对象能够被next()
函数调用并不断返回下一个数据,直到没有数据时抛出StopIteration
错误。能够把这个数据流看作是一个有序序列,但咱们却不能提早知道序列的长度,只能不断经过next()
函数实现按需计算下一个数据,因此Iterator
的计算是惰性的,只有在须要返回下一个数据时它才会计算。spa
Iterator
甚至能够表示一个无限大的数据流,例如全体天然数。而使用list是永远不可能存储全体天然数的。code
凡是可做用于for
循环的对象都是Iterable
类型;
凡是可做用于next()
函数的对象都是Iterator
类型,它们表示一个惰性计算的序列;
集合数据类型如list
、dict
、str
等是Iterable
但不是Iterator
,不过能够经过iter()
函数得到一个Iterator
对象。
Python的for
循环本质上就是经过不断调用next()
函数实现的。
经过列表生成式,咱们能够直接建立一个列表。可是,受到内存限制,列表容量确定是有限的。并且,建立一个包含100万个元素的列表,不只占用很大的存储空间,若是咱们仅仅须要访问前面几个元素,那后面绝大多数元素占用的空间都白白浪费了。
因此,若是列表元素能够按照某种算法推算出来,那咱们是否能够在循环的过程当中不断推算出后续的元素呢?这样就没必要建立完整的list,从而节省大量的空间。在Python中,这种一边循环一边计算的机制,称为生成器:generator。
要建立一个generator,有不少种方法。第一种方法很简单,只要把一个列表生成式的[]
改为()
,就建立了一个generator
什么是生成器?
能够理解为一种数据类型,这种数据类型自动实现了迭代器协议(其余的数据类型须要调用本身内置的__iter__方法),因此生成器就是可迭代对象
生成器分类及在python中的表现形式:(Python有两种不一样的方式提供生成器)
1.生成器函数:常规函数定义,可是,使用yield语句而不是return语句返回结果。yield语句一次返回一个结果,在每一个结果中间,挂起函数的状态,以便下次重它离开的地方继续执行
2.生成器表达式:相似于列表推导,可是,生成器返回按需产生结果的一个对象,而不是一次构建一个结果列表
为什么使用生成器之生成器的优势
Python使用生成器对延迟操做提供了支持。所谓延迟操做,是指在须要的时候才产生结果,而不是当即产生结果。这也是生成器的主要好处。
生成器小结:
1.是可迭代对象
2.实现了延迟计算,省内存啊
3.生成器本质和其余的数据类型同样,都是实现了迭代器协议,只不过生成器附加了一个延迟计算省内存的好处,其他的可迭代对象可没有这点好处,记住喽!!!
def lay_eggs(num): egg_list=[] for egg in range(num): egg_list.append('蛋%s' %egg) return egg_list yikuangdan=lay_eggs(10) #咱们拿到的是蛋 print(yikuangdan) def lay_eggs(num): for egg in range(num): res='蛋%s' %egg yield res print('下完一个蛋') laomuji=lay_eggs(10)#咱们拿到的是一只母鸡 print(laomuji) print(laomuji.__next__()) print(laomuji.__next__()) print(laomuji.__next__()) egg_l=list(laomuji) print(egg_l) #演示只能日后不能往前 #演示蛋下完了,母鸡就死了 母鸡下蛋的传说
综上已经对生成器有了必定的认识,下面咱们以生成器函数为例进行总结
优势一:生成器的好处是延迟计算,一次返回一个结果。也就是说,它不会一次生成全部的结果,这对于大数据量处理,将会很是有用。
1 #列表解析 2 sum([i for i in range(100000000)])#内存占用大,机器容易卡死 3 4 #生成器表达式 5 sum(i for i in range(100000000))#几乎不占内存
优势二:生成器还能有效提升代码可读性
#求一段文字中,每一个单词出现的位置 def index_words(text): result = [] if text: result.append(0) for index, letter in enumerate(text, 1): if letter == ' ': result.append(index) return result print(index_words('hello alex da sb')) 不使用迭代器
#求一段文字中每一个单词出现的位置 def index_words(text): if text: yield 0 for index, letter in enumerate(text, 1): if letter == ' ': yield index g=index_words('hello alex da sb') print(g) print(g.__next__()) print(g.__next__()) print(g.__next__()) print(g.__next__()) print(g.__next__())#报错 使用迭代器
这里,至少有两个充分的理由说明 ,使用生成器比不使用生成器代码更加清晰:
这个例子充分说明了,合理使用生成器,可以有效提升代码可读性。只要你们彻底接受了生成器的概念,理解了yield语句和return语句同样,也是返回一个值。那么,就可以理解为何使用生成器比不使用生成器要好,可以理解使用生成器真的可让代码变得清晰易懂。
注意事项:生成器只能遍历一次(母鸡一辈子只能下必定数量的蛋,下完了就死掉了)
人口信息.txt文件内容 {'name':'北京','population':10} {'name':'南京','population':100000} {'name':'山东','population':10000} {'name':'山西','population':19999} def get_provice_population(filename): with open(filename) as f: for line in f: p=eval(line) yield p['population'] gen=get_provice_population('人口信息.txt') all_population=sum(gen) for p in gen: print(p/all_population) 执行上面这段代码,将不会有任何输出,这是由于,生成器只能遍历一次。在咱们执行sum语句的时候,就遍历了咱们的生成器,当咱们再次遍历咱们的生成器的时候,将不会有任何记录。因此,上面的代码不会有任何输出。 所以,生成器的惟一注意事项就是:生成器只能遍历一次。 人口信息