迭代器

可迭代协议和迭代器协议

  • 可迭代协议

    只要含有__iter__方法的对象都是可迭代的

  • 迭代器协议

    内部含有__next__和__iter__方法的就是迭代器

  • 关系

    1.可以被for循环的都是可迭代的

    2.可迭代的内部都有__iter__函数

    3.只要是迭代器,一定可迭代

    4.可迭代的对象使用__iter__方法就可以得到一个迭代器

    5.迭代器中的__next__方法可以一个一个的获取值

例子

  • 判断是否是可迭代类型

    可以被for循环的类型都有__iter__()函数

    dir()函数可以返回类型里面所有函数名称

     print('__iter__' in dir(int))  # result:False
     print('__iter__' in dir(bool))  # result:False
     print('__iter__' in dir(list))  # result:True
     print('__iter__' in dir(dict))  # result:True
     print('__iter__' in dir(set))  # result:True
     print('__iter__' in dir(tuple))  # result:True
     print('__iter__' in dir(enumerate([])))  # result:True
     print('__iter__' in dir(range(1)))  # result:True
  • 判断是否是迭代器

    isinstance()函数可以判断一个对象是否是指定类型的实例

    import collections
    
    print(isinstance([], collections.Iterator))  # result: False
    print(isinstance([], collections.Iterable))  # result: True
    print(isinstance([].__iter__(), collections.Iterator))  # result:True
  • 通过__next__()函数取值

     num_list = [1, 2, True, 5, 'hello']
     print(type(num_list.__iter__()))  # result:<class 'list_iterator'>
     iterator = num_list.__iter__()
     print(iterator.__next__())  # result:1
     print(iterator.__next__())  # result:2
     print(iterator.__next__())  # result:True
     print(iterator.__next__())  # result:5
     print(iterator.__next__())  # result:hello
     print(iterator.__next__())  # 抛异常StopIteration

    当迭代器的next()方法取不到值时会抛一个StopIteration异常

迭代器的好处

1.从容器类型对象中一个一个的取值,会把所有的值都取到

2.节省内存空间(迭代器并不会在内存中再占用一大块内存,而是随着循环 每次生成(next())一个)

生成器函数

定义

只要含有yield关键字的函数都是生成器函数

   注:yield不能和return共用且需要写在函数内

例子

  • 生成器函数的执行

    执行之后会得到一个生成器作为返回值

     import collections;
    
     def generator():
         yield 'a'
         yield 'b'
    
     result = generator()
     print(type(result))  # <class 'generator'>
     print(isinstance(result, collections.Iterable))  # True
     print(isinstance(result, collections.Iterator))  # True
     print(result.__next__())  # a
     print(result.__next__())  # b

    通过第10行可以看出,生成器也是迭代器

  • 监听文件输入

     def listen(filename):
         f = open(filename, encoding='utf-8')
         while True:
             line = f.readline()
             if line.strip():
                 yield line.strip()
    
     listenr = listen('generator_file.txt')
     for line in listenr:
         print('your input:', line)

生成器进阶

send()函数使用

生成器中send()函数可以让上一个执行的yied接收一个值

 def num_generator():
     i = 1
     num = yield i
     i += num
     yield i

 generator = num_generator()

移动平均数例子

 def avg_num():
     sum = 0
     count = 0
     avg = 0
     while True:
         num = yield avg
         sum += num
         count += 1
         avg = sum / count

 ave_generator = avg_num()
 ave_generator.__next__()
 print(ave_generator.send(10))  # 10.0
 print(ave_generator.send(20))  # 15.0
 print(ave_generator.send(30))  # 20.0
 print(ave_generator.send(60))  # 30.0

面试题

  • 写出下面代码的输出内容

    def add(n, i):
        return n + i
    
    def test():
        for i in range(4):
            yield i
    
    g = test()
    for n in [1, 10]:
        g = (add(n, i) for i in g)
    print(list(g))
  • 分析&结果
      # 循环可拆成如下代码
      n = 1
      g = (add(n, i) for i in g)
      n = 10
      # g = (add(n, i) for i in g(指向第3行的g))
      # g = (add(n, i) for i in (add(n, i) for i in g(第三行的g指向test())))
      g = (add(n, i) for i in (add(n, i) for i in test()))
      # 真正执行的时候 也就是list(g)的时候 n=10 将n替换为10结果如下:
      g = (add(10, i) for i in (add(10, i) for i in (0, 1, 2, 3)))
      g = (add(10, i) for i in (10, 11, 12, 13))
      g = (20, 21, 22, 23)
      print(list(g))

推导式

列表

 num_list = [1, 2, 3, 4]
 str_list = ['数字{}'.format(i) for i in num_list]
 print(str_list)  # ['数字1', '数字2', '数字3', '数字4']

字典

反转dic的key和value的顺序

dic = {1: 'a', 2: 'b', 3: 'c'}
reverse_dic = {dic[key]: key for key in dic}
print(reverse_dic)  # {'a': 1, 'b': 2, 'c': 3}

集合

输出集合中元组里面大于10的元素

 num_set = {(3, 4, 1, 5, 77, 5), (2, 3, 4, 56, 77, 6, 34)};
 result_set = {i for lst in num_set for i in lst if i > 10}
 print(result_set)  # {56, 34, 77}

python基础(9)-迭代器&生成器函数&生成器进阶&推导式的更多相关文章

  1. (转)python基础之迭代器协议和生成器(一)

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

  2. python基础之 迭代器回顾,生成器,推导式

    1.迭代器回顾 可迭代对象:Iterable 可以直接作用于for循环的对象统称为可迭代对象:Iterable.因为可迭代对象里面存在可迭代协议,所以才会被迭代 可迭代对象包括: 列表(list) 元 ...

  3. python基础知识15---三元表达式、列表推导式、生成器表达式、递归、匿名函数、内置函数

    阅读目录 一 三元表达式.列表推导式.生成器表达式 二 递归与二分法 三 匿名函数 四 内置函数 五 阶段性练习 一. 三元表达式.列表推导式.生成器表达式 1 三元表达式 name=input('姓 ...

  4. python基础之迭代器协议和生成器

    迭代器和生成器补充:http://www.cnblogs.com/luchuangao/p/6847081.html 一 递归和迭代 略 二 什么是迭代器协议 1.迭代器协议是指:对象必须提供一个ne ...

  5. python基础之迭代器协议和生成器(二)

    一.什么是迭代器: 迭代是Python最强大的功能之一,是访问集合元素的一种方式. 迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束. 迭代器是一个可以记住遍历的位置的对象. 迭代器的 ...

  6. python基础之迭代器协议和生成器(一)

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

  7. python 生成器 和生成器函数 以及各种推导式

    一.生成器    本质就是迭代器. 我们可以直接执⾏__next__()来执⾏ 以下⽣成器 一个一个的创建对象 创建生成器的方式: 1.生成器函数 2.通过生成器 表达式来获取生成器 3.类型转换(看 ...

  8. day 12 生成器和生成器函数以及各种推导式

    一.生成器    本质就是迭代器. 我们可以直接执⾏__next__()来执⾏ 以下⽣成器 一个一个的创建对象 创建生成器的方式: 1.生成器函数 2.通过生成器 表达式来获取生成器 3.类型转换(看 ...

  9. python基础知识-8-三元和一行代码(推导式)

    python其他知识目录 1.三元运算(三目运算) 三元运算符就是在赋值变量的时候,可以直接加判断,然后赋值格式:[on_true] if [expression] else [on_false]re ...

  10. Python基础之列表深浅复制和列表推导式

    一.列表深浅复制: 浅拷贝内存图如下: 深拷贝内存图如下: 二.列表推导式: 实例: """ 列表推导式 练习:exercise01 """ ...

随机推荐

  1. 一个手写的 http client

    public class HTTPClient { public static final String GET = "GET"; public static final Stri ...

  2. java调用删除文件的方法删除文件,却删除不干净

    场景: 程序中在做数据下载时,生成了一个临时文件夹.夹子里面有一些txt和其他格式文件. 数据下载完毕后,需要删除这个临时文件夹,但是一直删除不干净,总会有一下文件残留. 网搜到了这个问题的原因: 内 ...

  3. [Android] 基于 Linux 命令行构建 Android 应用(三):构建流程

    Android 应用的构建过程就是将 Android 项目中的文件和资源进行编译和打包,最后将结果输出到 .apk 文件..apk 文件是保存二进制文件的容器,它包含了运行安卓应用所需的所有信息,例如 ...

  4. Linux设备驱动剖析之Input(二)

    分别是总线类型.厂商号.产品号和版本号. 1156行,evbit,设备支持的事件类型的位图,每一位代表一种事件,比如EV_KEY.EV_REL事件等等.BITS_TO_LONGS(nr)是一个宏,假设 ...

  5. A - ACM Rank Table

    ACM contests, like the one you are participating in, are hosted by the special software. That softwa ...

  6. pycharm更新之后显示问题

    pycharm更新之后显示问题 在新版pycharm中等号和其他符号会连在一块,下面是解决方法 添加公众号:

  7. datagrid---columns列

    { field:"city", //字段名,从后台传来的要一致 title:"城市",//列的标题文字. width:,//列的宽度 formatter:fun ...

  8. 一块移动硬盘怎样兼容Mac和Windows系统,并且可以在time machine上使用

    Mac的内存通常不是很大,加上使用Mac的time machine备份的话,是需要外置U盘或硬盘的.U盘存储空间较小,大家也不会去分区,直接拿来用在Mac上或者Windows上就可以了.所以这里只说硬 ...

  9. jdbc-------JDBCUtil类 工具类

    jdbcutil 主要处理的是 连接数据库, 和关闭各个流 1, 数据库连接的配置信息: mysql.properties (在工程的目录下)个人配置 url=jdbc:mysql://localho ...

  10. hung_task_timeout_secs和blocked for more than 120 seconds的解决方法

    Linux系统出现hung_task_timeout_secs和blocked for more than 120 seconds的解决方法 Linux系统出现系统没有响应. 在/var/log/me ...