迭代器详解

Python三大器之一

                                                                                            迭代器详解


1.问题由来:

            类似于遍历一个数组,字符串 ,元组,字典一样,可以遍历一个对象中的属性存储的列表吗?

            回答:不能

for  e  in range(5):
    print(e)  #  打印出来 1 2 3 4 5                                          【成功执行】
class ClassMate(object):
    def __init__(self):
        self.studentName = list()

    def add(self, name):
        self.studentName.append(name)

mate = ClassMate()
mate.add('张三')
mate.add('李四')
mate.add('王五')
for name in mate:
    print(name)                                          【报错,执行抛出异常】                                

以上遍历类的操作报错,抛出异常  TypeError: 'ClassMate' object is not iterable。

2.解决办法

        (1)理解异常 ->[TypeError: 'ClassMate' object is not iterable],这提示的是该对象不是可迭代对象。

            在Python中,一个对象中包含 __iter__魔法方法,就可以定为可迭代对象。

        (2)添加__iter__魔法方法到类中

            可以看到依然出现了问题,提示的是迭代器返回了不是迭代器类型的NoneType类型的对象,

简而言之就__iter__要返回一个迭代器类型的对象

            (3)接着返回一个迭代器类型的对象(迭代器必须包含__iter__ 和 __next__两个魔法方法)

                上图的黄色框中就是添加的返回迭代类型的对象和,迭代器类。当代码执行时,就可以看到每秒都会打印出返回值

看到这里,我们大概就明白了。迭代器的执行就是每次返回__next__返回的结果。那我们对iterOp类中的__next__方法改造一下,

这里基本可以遍历对象中指定的内容了。但是遍历结束后提交的是None,这里特别说明,对于迭代而言,当捕获到StopIteration异常,迭代就会停止

所以在if之后的else之后,抛出StopIteration异常   ,就可以结束迭代。至此迭代器完成。

import time
class ClassMate(object):
    def __init__(self):
        self.studentName = list()
    def add(self,name):
        self.studentName.append(name)
    def __iter__(self):
        return iterOp(self)
class iterOp(object):
    def __init__(self,obj):
        self.obj = obj
        self.count = 0
    def __iter__(self):
        pass
    def __next__(self):
        # self.obj.studentName[0]
        time.sleep(1)
        if self.count < len(self.obj.studentName):
            res = self.obj.studentName[self.count]
            self.count += 1
            return res
        else:
            raise StopIteration

mate = ClassMate()
mate.add('张三')
mate.add('李四')
mate.add('王五')
for name in mate:    
    print(name)

2.代码简化

通过上面迭代器必要组件的分析,包含有__iter__,__next__,方法的类都可以看做迭代器。所以讲可以将迭代器类简化到目标类中。

import time
class ClassMate(object):
    def __init__(self):
        self.count = 0    #  计数器初始化
        self.studentName = list()
    def add(self,name):
        self.studentName.append(name)
    def __iter__(self):
        return self                # 返回迭代器对象,self就是对象的引用。(自身此时包含__iter__和__next__方法,自己也
                                   #   就是迭代器)
    def __next__(self):
        # self.obj.studentName[0]
        time.sleep(1)
        if self.count < len(self.studentName):
            res = self.studentName[self.count]
            self.count += 1
            return res
        else:
            raise StopIteration
mate = ClassMate()
mate.add('张三')
mate.add('李四')
mate.add('王五')
for name in mate:    
    print(name)

3.迭代器总结

  • for 遍历的语法执行分析

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值