小猿圈python入门之迭代器和生成器

迭代器协议和生成器

迭代器协议:

1、迭代器协议是指:对象必须提供一个next()方法,执行该方法要么返回迭代中的下一项,要么引起一个StopIteration异常,以终止迭代(只往后走,不能往前退----》模拟人更新换代)

2、可迭代对象:实现可迭代协议的对象。(对象内部定义一个__iter__()方法)

3、协议是一种约定,可迭代对象实现了可迭代协议,python的内部工具(如for sum max min...函数)使用迭代器 协议访问对象。

for循环机制:

for循环的本质:循环所有对象,全部都使用迭代器协议。

for循环就是基于迭代器协议提供了一个统一的可遍历所有对象的方法,即在遍历之前,先调用对象的__iter__()方法将其转换为一个迭代器,然后使用迭代器协议去实现循环访问,这样所有对象都可以通过for循环来遍历了。

列表、字符串、元组、字典、集合、文件对象等本质上都不是可迭代对象,在使用for循环时内部先调用他们内部的__iter__方法,使它们变为一个可迭代对象,然后使用可迭代对象的next(可迭代对象)方法或者 对象.__next__(),依次循环元素,当元素循环完时,会触发StopIteration异常,for循环会捕捉到这种异常,终止迭代。

访问方式:下标方式、迭代器协议、for循环:

# _*_ encoding:utf-8 _*_

__author__ = 'listen'

__date__ = '2018/11/19 22:10'

# li=[1,2,3,4,5]  #通过下标索引方式取  只试用于有序的 列表 元组 字符串  不适用无序字典 集合

# print(li[0])  #1

# print(li[1])  #2

# print(li[2])  #3

 迭代器协议:

#通过迭代器访问

# li=[1,2,3,4,5,6]

# iter_li=li.__iter__()  #通过__iter__()方法 生成一个可迭代对象

# print(iter_li.__next__())  #1  可迭代对象就有__next__()方法  一个一个取值

# print(iter_li.__next__())  #2

# print(iter_li.__next__())  #3

# print(iter_li.__next__())

# print(iter_li.__next__())

# print(iter_li.__next__())

# print(iter_li.__next__())  #StopIteration,超出边界报错

for循环访问:

#for 循环访问的本质==遵循迭代器访问方式,先调用iter_l=li.__iter__()方法,然后依次执行iter_l.__next__(),直到for循环捕捉到StopIteration终止循环

# li=[1,2,3,4,5]

# for i in li:

#    print(i)

#用while去模拟for循环做的事情

# li=[1,2,3,4,5]

# index=0

# while index

#    print(li[index])

#    index+=1

#next()和__next__()方法是相同的

# li=[1,2,3]

# iter_li=li.__iter__()

# print(iter_li.__next__())  #1

# print(next(iter_li))  #2

生成器

定义:生成器类似于一种数据类型,这种数据类型自动实现迭代器协议(其他的数据类型都需要调用自身的__iter__方法),所以生成器就是可迭代对象。

1、生成器函数:和常规函数没什么区别,但使用yield语句返回结果 而不是return,yield语句一次返回一个结果,但每个结果中间,挂起函数的状态,以便下次从它离开的地方继续执行。

2、生成器表达式:类似于列表推导式,但是生成器返回按需产生的结果的一个对象(类似内存的一个地址),而不是一次构建结果的列表,按需取对象。

使用生成器的优点:

python使用生成器对延迟操作提了支持。所谓延迟操作,是指需要的时候才产生结果,而不是立即产生结果,节省内存,运行效率高,这是生成器的主要好处。

生成器表达式和列表表达式

三元表达式:f=if取出的结果 if条件  else  else结果

列表解析:s=[三元表达式],列表解析生成的是一个真实存在的列表,对于比较大的列表,消耗内存比较大。

count=[x for x in range(3)]

生成器表达式1:s=(三元表达式),区别于列表解析是 [] 到 ()

count=(x for x in range(3)) #生成器本身就是迭代器,遵循迭代器协议

#生成器----一种数据类型,这种数据类型自动实现迭代器协议(其他数据类型(字典 list tuple等等)都需要调用内置的__iter__()方法),所以生成器就是可迭代对象

#1、生成器函数  yield 代替return语句返回  可以调用多次

# def fun():

#    yield 1  #可迭代对象

#    yield 2

#    yield 3

# f=fun()

# print(f.__next__())  #1

# print(f.__next__())  #2

# print(f.__next__())  #3

# print(f.__next__()) ##执行只能一直往前走,不能往后,只能执行一次,执行完继续执行会触发StopIteration

#2、生成器表达式

#三元表达式  result=值1  if 条件  else 值2  列表表达式=[]是真实存在内存中的列表,对于大的列表特别消耗内存

# count=[x for x in range(3)]    #三元表达式

# count=(x for x in range(3))  #生成器

# sum(x for x in range(3)) ##sum函数是Python的内置函数,该函数使用迭代器协议访问对象,而生成器实现了迭代器协议,不用多此一举先生成列表

#鸡下蛋

laomuji=('鸡蛋%s'%i  for i in range(9) if i<5)

print(laomuji) # at 0x00000221029D1C50>  这是一个可迭代对象就不会占用很大的内存

print(laomuji.__next__())

print(laomuji.__next__())

print(laomuji.__next__())

print(laomuji.__next__())

print(laomuji.__next__())

print(next(laomuji))  #StopIteration抛异常

 生成器只能往后走,不能往前退,且只能执行一次。

# _*_ encoding:utf-8 _*_

__author__ = 'listen'

__date__ = '2018/11/20 21:14'

# #执行只能一直往前走,不能往后,只能执行一次

# def test():

#    for i in range(4):

#        yield i

# t=test()

# t1=(i for i in t)

# t2=(i for i in t1)

# print(list(t1))  #list()就是在执行next方法  for也是在执行next方法  [0, 1, 2, 3]

# print(list(t2))  #  []  t1的数据取完了,没有了,所以list(t2)为空

迭代器和生成器到这里就结束了,相信大家也学会了,有问题或者想学习成套视频,欢迎加入python自学交流QQ群:242719133,也可以给小编留言哦,相信我们一起学习,最后会掌握python这门网红技能,fighting~~

你可能感兴趣的:(小猿圈python入门之迭代器和生成器)