python之生成器与迭代器详解

1 可迭代对象(iterable

​ 如果一个对象具有 备__iter__()方法或者 __getitem__() 其中任何一个魔术方法的话,该对象就可以称为可迭代对象,所以python内置的对象如列表、元祖、字符串、字典都是可迭代的,即是可迭代对象,但不是迭代器。

# 调用可迭代对象的__iter__()方法可获得迭代器(iterator)
print(type([].__iter__()))
#<class 'list_iterator'>
2 迭代器(iterator)

​ 如果一个对象同时j具有有 __iter__()__next__() 魔术方法的话,这个对象就可以称为是迭代器。next__()方法是让对象可以通过 对象.__next()__ 的方式访问下一个元素。列表、元组、字典、字符串都是通过先调用__iter__()方法获取迭代器,然后通过返回的迭代器再调用__next()__方法访问对象的每个元素。

l1 = [1, 2, 3]  # l1为可迭代对象
print(dir(l1))
# [...'__iter__'...]

s1 = l1.__iter__()
print(dir(s1))  # s1为迭代器
# [...'__next__'...]

# 通过调用__next__()方法依次获取对象中的每个元素
print(s1.__next__())  # 1
print(s1.__next__())  # 2
print(s1.__next__())  # 3

​ 对象具有 __iter__() 方法才能被for循环调用,如下可以自己定义可迭代对象及迭代器,因此可以看出,迭代器一定是可迭代对象,但是可迭代对象不一定是迭代器。

# 自定义一个可迭代对象、且__iter__()方法返回的是对象本事(对象本事亦是一个迭代器)
class Test:
    def __init__(self, x):
        self.x = x

    def __iter__(self):
        return self

    def __next__(self):
        if self.x >= 10:
            raise StopIteration
        else:
            self.x += 1
            return self.x


if __name__ == '__main__':
    service = Test(1)
    print(service.__next__())  # 2
    print(service.__next__())  # 3
    print(service.__next__())  # 4
    print(service.__next__())  # 5

    for item in service:
        print(item, end=' ')
        # 6 7 8 9 10

​ 可迭代与迭代器按照内存来解释:

l = [1, 2, 3]
s = l.__iter__()
p = l.__iter__()

print(id(p))  # 2780542485792
print(id(s))  # 2780542485984

print(id(s.__iter__()))  # 2780542485984

print(l.__iter__().__next__())  # 1
print(l.__iter__().__next__())  # 1

​ for循环的原理:

1. 调用对象的__iter__()方法获取迭代器;
2. 调用迭代器的__next__()方法循环获取对象的下一个元素;
3. 遇到StopIteration异常停止迭代;
3 生成器(generator)

​ 在 Python 中,使用了 yield 的函数被称为生成器(generator)。

​ 跟普通函数不同的是,生成器是一个返回迭代器的函数,只能用于迭代操作,更简单点理解生成器就是一个迭代器。在调用生成器运行的过程中,每次遇到 yield 时函数会暂停并保存当前所有的运行信息,返回 yield 的值, 并在下一次执行 next() 方法时从当前位置继续运行。调用一个生成器函数,返回的是一个迭代器对象。

​ 如果一个函数包含 yield 关键字,这个函数就会变为一个生成器。生成器并不会一次返回所有结果,而是每次遇到 yield 关键字后返回相应结果,并保留函数当前的运行状态,等待下一次的调用。

def generate():
    """
    生成器函数
    """
    index = 10
    while index > 0:
        yield index
        index -= 1


generator = generate()

print(type(generator))  # <class 'generator'>

for i in generator:
    print(i, end=' ')
# 10 9 8 7 6 5 4 3 2 1

​ 生成器(generator) 自动实现了迭代器协议 ;

​ 迭代器协议:对象必须提供一个next方法,执行方法要么返回迭代器中的下一项,要么就引起一个StopIteration异常,以终止迭代(只能往后走,不能往前退) ;

​ 可迭代对象:实现了迭代器协议的对象(实现方式:对象内部定义了一个iter()方法);

case:应用生成器模式实现斐波拉切数列:

# F(n)=F(n-1)+F(n-2)
# 1,1,2,3,5,8,13...
def fei(n):
    index = 1
    pre = 0
    cur = 1
    while index <= n:
        yield cur
        index += 1
        cur, pre = cur + pre, cur
f = fei(9)
for item in f:
    print(item, end=' ')  # 1 1 2 3 5 8 13 21 34
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值