• Python学习第十天 迭代器 生成器 三元表达式


    一 迭代器

    一 迭代的概念

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

    2.可迭代对象:实现了迭代器协议的对象(如何实现:对象内部定义一个_iter_()方法)

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

    #迭代器即迭代的工具,那什么是迭代呢?
    #迭代是一个重复的过程,每次重复即一次迭代,并且每次迭代的结果都是下一次迭代的初始值
    while True: #只是单纯地重复,因而不是迭代
        print('===>') 
        
    l=[1,2,3]
    count=0
    while count < len(l): #迭代
        print(l[count])
        count+=1

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

    (字符串,列表,元组,字典,集合,文件对象)这些都不是迭代器对象,只不过在for循环中,调用了他们内部的_iter_方法,把他们变成了可迭代对象。然后,for循环调用可迭代对象的_next_方法去取值,而且for循环会捕捉Stopiteration异常,以终止迭代

    二 为何要有迭代器?什么是可迭代对象?什么是迭代器对象?

    #1、为何要有迭代器?
    对于序列类型:字符串、列表、元组,我们可以使用索引的方式迭代取出其包含的元素。但对于字典、集合、文件等类型是没有索引的,
    若还想取出其内部包含的元素,则必须找出一种不依赖于索引的迭代方式,这就是迭代器
    #2、什么是可迭代对象? 可迭代对象指的是内置有__iter__方法的对象,即obj.__iter__,如下 'hello'.__iter__ (1,2,3).__iter__ [1,2,3].__iter__ {'a':1}.__iter__ {'a','b'}.__iter__ open('a.txt').__iter__ #3、什么是迭代器对象? 可迭代对象执行obj.__iter__()得到的结果就是迭代器对象 而迭代器对象指的是即内置有__iter__又内置有__next__方法的对象 文件类型是迭代器对象 open('a.txt').__iter__() open('a.txt').__next__() #4、注意: 迭代器对象一定是可迭代对象,而可迭代对象不一定是迭代器对象

    三 迭代器对象的使用

    dic={'a':1,'b':2,'c':3}
    iter_dic=dic.__iter__() #得到迭代器对象,迭代器对象即有__iter__又有__next__,但是:迭代器.__iter__()得到的仍然是迭代器本身
    iter_dic.__iter__() is iter_dic #True
    
    print(iter_dic.__next__()) #等同于next(iter_dic)
    print(iter_dic.__next__()) #等同于next(iter_dic)
    print(iter_dic.__next__()) #等同于next(iter_dic)
    # print(iter_dic.__next__()) #抛出异常StopIteration,或者说结束标志
    
    #有了迭代器,我们就可以不依赖索引迭代取值了
    iter_dic=dic.__iter__()
    while 1:
        try:
            k=next(iter_dic)
            print(dic[k])
        except StopIteration:
            break
            
    #这么写太丑陋了,需要我们自己捕捉异常,控制next,python这么牛逼,能不能帮我解决呢?能,请看for循环

    四 for循环

    #基于for循环,我们可以完全不再依赖索引去取值了
    dic={'a':1,'b':2,'c':3}
    for k in dic:
    print(dic[k])
    
    #for循环的工作原理
    #1:执行in后对象的dic.__iter__()方法,得到一个迭代器对象iter_dic
    #2: 执行next(iter_dic),将得到的值赋值给k,然后执行循环体代码
    #3: 重复过程2,直到捕捉到异常StopIteration,结束循环

    五 迭代器的优缺点

    #优点:
      - 提供一种统一的、不依赖于索引的迭代方式
      - 惰性计算,节省内存
    #缺点:
      - 无法获取长度(只有在next完毕才知道到底有几个值)
      - 一次性的,只能往后走,不能往前退
     

    二 生成器

    一 什么是生成器

    可以理解为一种数据类型,这种数据类型自动实现了迭代器协议(其他数据类型需要调用自己内置的_iter_方法),所以生成器就是可迭代对象。

    生成器分类及在python中的表现形式:(python有两种不同的方式提供生成器)
    1.生成器函数:常规函数定义,但是,使用yield语句而不是return语句返回结果。yield语句一次返回一个结果,在每个结果中间,挂起函数的状态,以便下次从它离开的地方继续执行。
    2.生成器表达式:类似于列表推导,但是,生成器返回按需产生结果的一个对象,而不是一次构建一个结果列表
    为何使用生成器之生成器的优点:
    python使用生成器对延迟操作提供了支持。所谓延迟操作,是指在需要的时候才产生结果,而不是立即产生结果。这也是生成器的主要好处。
    生成器小结:
    1.是可迭代对象
    2.实现了延迟结算,省内存
    3.生成器本质和其他的数据类型给一样,都是实现了迭代器协议,只不过生成器附加了一个延迟计算省内存的好处,其余的可迭代对象可没有这点好处。

    #只要函数内部包含有yield关键字,那么函数名()的到的结果就是生成器,并且不会执行函数内部代码
    
    def func():
        print('====>first')
        yield 1
        print('====>second')
        yield 2
        print('====>third')
        yield 3
        print('====>end')
    
    g=func()
    print(g) #<generator object func at 0x0000000002184360>

     生产者消费者模型 

    import time
    def consumer(name):
        print('我是[%s],我准备开始吃包子了' %name)
        while True:
            baozi=yield
            time.sleep(1)
            print('%s 很开心的把【%s】吃掉了' %(name,baozi))
    
    def producer():
        c1=consumer('wupeiqi')
        c2=consumer('yuanhao_SB')
        c1.__next__()
        c2.__next__()
        for i in range(10):
            time.sleep(1)
            c1.send('包子 %s' %i)
            c2.send('包子 %s' %i)
    producer()
    #yield 3相当于return 控制的是函数的返回值
    #x=yield的另外一个特性,接受send传过来的值,赋值给x
    # def test():
    #     print('开始啦')
    #     firt=yield #return 1   first=None
    #     print('第一次',firt)
    #     yield 2
    #     print('第二次')
    #
    # t=test()
    # res=t.__next__() #next(t)
    # print(res)
    # # t.__next__()
    # # res=t.send(None)
    # res=t.send('函数停留在first那个位置,我就是给first赋值的')
    # print(res)

    二 生成器就是迭代器

    g.__iter__
    g.__next__
    #2、所以生成器就是迭代器,因此可以这么取值
    res=next(g)
    print(res)

    三 练习

    1、自定义函数模拟range(1,7,2)

    2、模拟管道,实现功能:tail -f access.log | grep '404'

    #题目一:
    def my_range(start,stop,step=1):
        while start < stop:
            yield start
            start+=step
    
    #执行函数得到生成器,本质就是迭代器
    obj=my_range(1,7,2) #1  3  5
    print(next(obj))
    print(next(obj))
    print(next(obj))
    print(next(obj)) #StopIteration
    
    #应用于for循环
    for i in my_range(1,7,2):
        print(i)
    
    #题目二
    import time
    def tail(filepath):
        with open(filepath,'rb') as f:
            f.seek(0,2)
            while True:
                line=f.readline()
                if line:
                    yield line
                else:
                    time.sleep(0.2)
    
    def grep(pattern,lines):
        for line in lines:
            line=line.decode('utf-8')
            if pattern in line:
                yield line
    
    for line in grep('404',tail('access.log')):
        print(line,end='')
    
    #测试
    with open('access.log','a',encoding='utf-8') as f:
        f.write('出错啦404
    ')

    一 三元表达式、列表推导式、生成器表达式

    一 三元表达式

    name=input('姓名>>: ')
    res='SB' if name == 'alex' else 'NB'
    print(res)

    二 列表推导式

    #1、示例
    egg_list=[]
    for i in range(10):
        egg_list.append('鸡蛋%s' %i)
    
    egg_list=['鸡蛋%s' %i for i in range(10)]
    
    #2、语法
    [expression for item1 in iterable1 if condition1
    for item2 in iterable2 if condition2
    ...
    for itemN in iterableN if conditionN
    ]
    类似于
    res=[]
    for item1 in iterable1:
        if condition1:
            for item2 in iterable2:
                if condition2
                    ...
                    for itemN in iterableN:
                        if conditionN:
                            res.append(expression)
    
    #3、优点:方便,改变了编程习惯,可称之为声明式编程

    三 生成器表达式

    #1、把列表推导式的[]换成()就是生成器表达式
    
    #2、示例:生一筐鸡蛋变成给你一只老母鸡,用的时候就下蛋,这也是生成器的特性
    >>> chicken=('鸡蛋%s' %i for i in range(5))
    >>> chicken
    <generator object <genexpr> at 0x10143f200>
    >>> next(chicken)
    '鸡蛋0'
    >>> list(chicken) #因chicken可迭代,因而可以转成列表
    ['鸡蛋1', '鸡蛋2', '鸡蛋3', '鸡蛋4',]
    
    #3、优点:省内存,一次只产生一个值在内存中

    四 声明式编程练习题

    1、将names=['egon','alex_sb','wupeiqi','yuanhao']中的名字全部变大写

    2、将names=['egon','alex_sb','wupeiqi','yuanhao']中以sb结尾的名字过滤掉,然后保存剩下的名字长度

    3、求文件a.txt中最长的行的长度(长度按字符个数算,需要使用max函数)

    4、求文件a.txt中总共包含的字符个数?思考为何在第一次之后的n次sum求和得到的结果为0?(需要使用sum函数)

    5、思考题

    with open('a.txt') as f:
        g=(len(line) for line in f)
    print(sum(g)) #为何报错?

    6、文件shopping.txt内容如下

    mac,20000,3
    lenovo,3000,10
    tesla,1000000,10
    chicken,200,1

    求总共花了多少钱?

    打印出所有商品的信息,格式为[{'name':'xxx','price':333,'count':3},...]

    求单价大于10000的商品信息,格式同上

    #题目一
    names=['egon','alex_sb','wupeiqi','yuanhao']
    names=[name.upper() for name in names]
    
    #题目二
    names=['egon','alex_sb','wupeiqi','yuanhao']
    names=[len(name) for name in names if not name.endswith('sb')]
    
    #题目三
    with open('a.txt',encoding='utf-8') as f:
        print(max(len(line) for line in f))
    
    #题目四
    with open('a.txt', encoding='utf-8') as f:
        print(sum(len(line) for line in f))
        print(sum(len(line) for line in f)) #求包换换行符在内的文件所有的字符数,为何得到的值为0?
        print(sum(len(line) for line in f)) #求包换换行符在内的文件所有的字符数,为何得到的值为0?
    
    #题目五(略)
    
    #题目六:每次必须重新打开文件或seek到文件开头,因为迭代完一次就结束了
    with open('a.txt',encoding='utf-8') as f:
        info=[line.split() for line in f]
        cost=sum(float(unit_price)*int(count) for _,unit_price,count in info)
        print(cost)
    
    
    with open('a.txt',encoding='utf-8') as f:
        info=[{
            'name': line.split()[0],
            'price': float(line.split()[1]),
            'count': int(line.split()[2]),
        } for line in f]
        print(info)
    
    
    with open('a.txt',encoding='utf-8') as f:
        info=[{
            'name': line.split()[0],
            'price': float(line.split()[1]),
            'count': int(line.split()[2]),
        } for line in f if float(line.split()[1]) > 10000]
        print(info)
  • 相关阅读:
    事务的特性(ACID)
    网络代理
    防止SpringMVC拦截器拦截js等静态资源文件
    Tomcat配置虚拟目录
    SpringMVC总结(SSM)
    Spring声明式事务总结
    Linux中MySQL忽略表中字段大小写
    MySQL之sql文件的导入导出
    MyBatis总结
    Linux网络
  • 原文地址:https://www.cnblogs.com/sundanceS/p/12594723.html
Copyright © 2020-2023  润新知