函数进阶--生成器与迭代器

列表生成式

列表生成式就是将一些简单的代码合并成一条代码来写

# 需要将列表[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]各自加1
#普通会这样写
a = [1,3,4,6,7,7,8,9,11]
for index,i in enumerate(a):
    a[index] +=1
print(a)
#用列表生成式
a = [i+1 for i in range(10)]
print(a)

生成器

带有 yield 关键字的的函数在 Python 中被称之为 generator(生成器)。Python 解释器会将带有 yield 关键字的函数视为一个 generator 来处理。一个函数或者子程序都只能 return 一次,但是一个生成器能暂停执行并返回一个中间的结果 —— 这就是 yield 语句的功能 : 返回一个中间值给调用者并暂停执行。

生成器有两种创建方式:

  • 列表生成式,以()形式
  • 函数

yield、return、next

  • yield 返回数据,并冻结当前的执行过程
  • return 返回数据,并结束当前函数
  • next 是唤醒冻结函数的执行过程,继续执行,直到遇到下一个yield
  • 当函数里有了yield之后,函数名加()就可以变成生成器
  • return在生成器里代表生成器的中止,会直接报错

通过生成器来实现日志文件

通过生成器写一个日志调用方法, 支持以下功能
根据指令向屏幕输出日志
根据指令向文件输出日志
根据指令同时向文件&屏幕输出日志
以上日志格式如下
代码:

import time
cont = 0
def logger(filename,channel='file'):
    """
    日志方法
    :param filename: log filename
    :param channel: 输出的目的地,屏幕(terminal),文件(file),屏幕+文件(both)
    :return:
    """
    global cont
    while True:
        cont += 1
        msg = yield
        a = time.strftime('%Y-%m-%d %X')
        info = '%s[%s]%s'%(a,cont,msg)
        if channel == 'file':
            with open(filename,'a',encoding='utf-8') as f:
                f.write('\n%s'%info)
        elif channel == 'terminal':
            print(info)
        elif channel == 'both':
            with open(filename,'a',encoding='utf-8') as f:
                f.write('\n%s'%info)
            print(info)
a = logger(filename='a.txt',channel='both')
next(a)
a.send('user alex login success')

迭代器

判断一个对象是否是可迭代对像可以用isinstance()来进行判断

>>> from collections import Iterable
>>> isinstance([], Iterable)
True
>>> isinstance({}, Iterable)
True
>>> isinstance('abc', Iterable)
True
>>> isinstance((x for x in range(10)), Iterable)
True
>>> isinstance(100, Iterable)
False

可以被next()函数调用并不断返回下一个值的对象称为迭代器:Iterator。
可以使用isinstance()判断一个对象是否是Iterator(迭代器)对象

>>> from collections import Iterator
>>> isinstance((x for x in range(10)), Iterator)
True
>>> isinstance([], Iterator)
False
>>> isinstance({}, Iterator)
False
>>> isinstance('abc', Iterator)
False

生成器都是Iterator对象,但list、dict、str虽然是Iterable,却不是Iterator。
把list、dict、str等Iterable变成Iterator可以使用iter()函数:

>>> isinstance(iter([]), Iterator)
True
>>> isinstance(iter('abc'), Iterator)
True
  • 凡是可作用于for循环的对象都是Iterable(可迭代对象)类型;
  • 凡是可作用于next()函数的对象都是Iterator类型,它们表示一个惰性计算的序列;
  • 集合数据类型如list、dict、str等是Iterable但不是Iterator,不过可以通过iter()函数获得一个Iterator对象。

猜你喜欢

转载自www.cnblogs.com/yjiu1990/p/9057438.html