Python三大器之生成器

生成器初识

什么是生成器


  生成器本身属于迭代器。继承了迭代器的特性,惰性求值,占用内存空间极小。

为什么要有生成器


  我们想使用迭代器本身惰性求值的特点创建出一个可以容纳百万级别的迭代器。(节省内存)又不想通过调用可迭代对象下的__iter__方法来创建这样的迭代器。(未经过优化的可迭代对象本身就占据内存,如listtupledictsetstr等) 这个时候就需要用到生成器。

怎么用生成器


  定义生成器的方式有两种。

  1.使用生成器表达式(本章不做介绍)

  2.使用生成器函数

  斐波拉契数列的创建:

  1. # ==== 使用yield创建生成器 ====

  2. import array # 数组,只能存放单一数据类型。如果要存放同一类型的数据,它比列表更好用

  3. s = array.array("I")
  4. def fblq(n):
  5. x,y,z = 0,0,1
  6. while x < n:
  7. yield y # 相当于return,暂停在此处。再次调用时继续执行下面的代码
  8. y,z=z,y+z
  9. x += 1
  10. g = fblq(10)
  11. print(g) # 现在的g是一个生成器对象. <generator object fblq at 0x000001F2C6EDE350>

  12. for i in g:
  13. print(i) # 取出一个,计算一个。
  14. s.append(i)
  15. print(s) # array('I', [0, 1, 1, 2, 3, 5, 8, 13, 21, 34])

惰性求值:需要用的时候经过内部计算取出一个。而不是管你目前用不用得到全部给你取出来。

生成器总结

优点


  节省内存。不用通过创建一个未经优化的可迭代对象再调用其__iter__方法创建迭代器。

缺点


  由于生成器本身就是属于迭代器。故缺点是只能使用一次,当值全部取出后该生成器对象意味着死亡。 如果生成器要取中间的值,只能通过一个一个的迭代过去。不能直接取出中间的值。

  故:Python对于listdict等数据类型为何不直接采取引用迭代器的方式呢?这是因为Python考虑到其还有其他的取值方式。如indexkey等等取值要比遍历取值更为方便。

  引用迭代器(经过优化的可迭代对象):

  可迭代对象本身并不存储任何值,for循环该可迭代对象时实际上就是生成一个迭代器,再通过该专属迭代器的__next__方法内部计算出需要的值并且返回。   

  这么做的方式在于不能通过index取值,但是极大节省内存空间。采用引用迭代器方式的数据类型有很多,比如:keys(),values(),items(),range()

扩展:生成器与协程

  生成器由于具有挂起当前函数状态的特性,所以可以有很多骚操作玩法,也间接的让协程成为可能。我们可以让一个生成器函数做不同的事情,根据不同的情况返回不同的结果。

  需要注意,yield本身具有return返回值的功能。并且还有接收值的功能。

  yield返回值所接收的对象将获得两个方法:

  send() ---> 向yield发送一个任意类型参数。

  close() ---> 当使用该方法后,将不再具有send()方法。

  1. # ==== 生成器的send与close ====

  2. def dog():
  3. # 等待send(None)或者next(host)执行。
  4. print("dog的绳子被主人拉上了..")
  5. a = yield "dog饿了" # 返回值,相当于狗对人说的话。a相当于外部第二次send进来的值
  6. print("dog吃了一坨", a)
  7. b = yield "dog渴了"
  8. print("dog喝了一口", b)
  9. yield "dog吃饱喝足了"


  10. host = dog()
  11. msg = host.send(None) # 第一次启动必须是None。或者使用next()开始生成器的执行。
  12. print(msg)
  13. msg = host.send("冰淇淋") # 对于send来说。内部有几个yield外部就该有几个send
  14. print(msg)
  15. msg = host.send("82年的雪碧")
  16. print(msg)

  17. # ==== 执行结果 ====

  18. """
  19. dog的绳子被主人拉上了..
  20. dog饿了
  21. dog吃了一坨 冰淇淋
  22. dog渴了
  23. dog喝了一口 82年的雪碧
  24. dog吃饱喝足了
  25. """

生成器的send与close

扩展:函数状态挂起底层原理

  Python中生成器函数是一个非常牛逼的东西。它可以让函数挂起状态,那么底层到底是怎么实现的呢?

  原文链接:https://www.cnblogs.com/traditional/p/9221680.html

  另外推荐深度好文:https://zhuanlan.zhihu.com/p/37109168

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3. # author:love_cat



  4. # python的函数是如何工作的

  5. # 比方说我们定义了两个函数


  6. def foo():
  7. bar()


  8. def bar():
  9. pass


  10. # 首先python解释器(python.exe)会用一个叫做PyEval_EvalFrameEx()的C语言函数去执行foo,所以python的代码是运行在C程序之上的
  11. # 当运行foo函数时,会首先创建一个栈帧(stack frame),表示函数调用栈当中的某一帧,相当于一个上下文,函数要在对应的栈帧上运行。
  12. # 正所谓python一切皆对象,栈帧也是一个对象
  13. # python虽然是解释型语言,但在解释之前也要进行一次预编译,编译成字节码对象,然后在对应的栈帧当中运行

  14. # 关于python的编译过程,我们可以是dis模块查看编译后的字节码是什么样子
  15. import dis
  16. print(dis.dis(foo))
  17. # 程序运行结果
  18. '''
  19. 0 LOAD_GLOBAL 0 (bar)
  20. CALL_FUNCTION 0
  21. POP_TOP
  22. LOAD_CONST 0 (None)
  23. RETURN_VALUE
  24. None
  25. '''
  26. # 首先LOAD_GLOBAL,把bar这个函数给load进来
  27. # 然后CALL_FUNCTION,调用bar函数的字节码
  28. # POP_POP,从栈的顶端把元素打印出来
  29. # LOAD_CONST,我们这里没有return,所以会把None给load进来
  30. # RETURN_VALUE,把None给返回
  31. '''
  32. 以上是字节码的执行过程
  33. '''

  34. # 过程就是:
  35. '''
  36. 1.先预编译,得到字节码对象
  37. 2.python解释器去解释字节码
  38. 3.当解释到foo函数的字节码时,会为其创建一个栈帧
  39. 4.然后调用C函数PyEval_EvalFrameEx()在foo对应的栈帧上执行foo的字节码,参数就是foo对应的栈帧对象
  40. 5.当遇到CALL_FUNCTION,也就是在foo中执行到bar的字节码时,会继续为其创建一个栈帧
  41. 6.然后把控制权交给新创建的栈帧对象,在bar对应的栈帧中运行bar的字节码
  42. '''

  43. # 我们看到目前已经有两个栈帧了,这不是关键。关键所有的栈帧都分配在堆的内存上,而不是栈的内存上
  44. # 堆内存有一个特点,如果你不去释放,那么它就一直待在那儿。这就决定了栈帧可以独立于调用者存在
  45. # 即便调用者不存在,或者函数退出了也没有关系,因为它始终在内存当中。只要有指针指向它,我们就可以对它进行控制
  46. # 这个特性决定了我们对函数的控制会相当精确。
  47. # 我们可以改写这个函数
  48. # 在此之前,我们要引用一个模块inspect,可以获取栈帧
  49. import inspect

  50. frame = None
  51. def foo():
  52. bar()

  53. def bar():
  54. global frame
  55. frame = inspect.currentframe() # 将获取到的栈帧对象赋给全局变量

  56. foo()
  57. # 此时函数执行完毕,但是我们依然可以拿到栈帧对象
  58. # 栈帧对象一般有三个属性
  59. # 1.f_back,当前栈帧的上一级栈帧
  60. # 2.f_code,当前栈帧对应的字节码
  61. # 3.f_locals,当前栈帧所用的局部变量

  62. print(frame.f_code)
  63. print(frame.f_code.co_name)
  64. '''
  65. <code object bar at 0x000000000298C300>
  66. bar
  67. '''
  68. # 可以看出,打印的是我们bar这个栈帧


  69. # 之前说过,栈帧可以独立于调用方而存在
  70. # 我们也可以拿到foo的栈帧,也就是bar栈帧的上一级栈帧
  71. foo_frame = frame.f_back
  72. print(foo_frame.f_code)
  73. print(foo_frame.f_code.co_name)
  74. '''
  75. <code object foo at 0x000000000239C8A0>
  76. foo
  77. '''
  78. # 我们依然可以拿到foo的栈帧

  79. # 总结一下:就是有点像递归。遇见新的调用,便创建一个新的栈帧,一层层地创建,然后一层层地返回

Python的普通函数与栈帧

  

  这种在函数内调用另一个函数的方式类似于递归,我们可以看一张图:

  1. # 我们之前说了,栈帧是分配在堆内存上的
  2. # 正是因为如此,生成器才有实现的可能

  3. # 我们定义一个生成器
  4. def gen_func():
  5. yield 123
  6. name = "satori"
  7. yield 456
  8. age = 18
  9. return "i love satori" # 注意在早期的版本中生成器是不允许有返回值的,但在后来的版本中,允许生成器具有返回值

  10. # python解释之前,也进行预编译,在编译的过程中,发现有yield,就已经被标记为生成器了

生成器与栈帧

  1. def gen_func():
  2. yield 123
  3. name = "satori"
  4. yield 456
  5. age = 18
  6. return "i love satori"
  7.  
  8. import dis
  9. gen = gen_func()
  10. print(dis.dis(gen))
  11. '''
  12. 0 LOAD_CONST 1 (123)
  13. YIELD_VALUE
  14. POP_TOP
  15. 6 LOAD_CONST 2 ('satori')
  16. STORE_FAST 0 (name)
  17. 10 LOAD_CONST 3 (456)
  18. YIELD_VALUE
  19. POP_TOP
  20. 16 LOAD_CONST 4 (18)
  21. STORE_FAST 1 (age)
  22. 20 LOAD_CONST 5 ('i love satori')
  23. RETURN_VALUE
  24. None
  25. '''
  26.  
  27. # 可以看到,结果中有两个yield,因为我们的函数中有两个yield
  28. # 最后的LOAD_CONST后面的('i love satori'),表示我们的返回值
  29. # 最后RETURN_VALUE
  30.  
  31. # 前面的图也解释了,gi_frame的f_lasti会记录最近的一次执行状态,gi_locals会记录当前的局部变量
  32. print(gen.gi_frame.f_lasti)
  33. print(gen.gi_frame.f_locals)
  34. '''
  35. -1
  36. {}
  37. '''
  38. # 我们创建了生成器,但是还没有执行,所以值为-1,当前局部变量也为空
  39.  
  40. # 我们next一下
  41. next(gen)
  42. print(gen.gi_frame.f_lasti)
  43. print(gen.gi_frame.f_locals)
  44. '''
  45. {}
  46. '''
  47. # 我们发现数字是2,所以指向第二行,YIELD_VALUE,yield的值就是123
  48. # 此时局部变量依旧为空
  49.  
  50. # 继续next,会执行到第二个yield的位置
  51. next(gen)
  52. print(gen.gi_frame.f_lasti)
  53. print(gen.gi_frame.f_locals)
  54. '''
  55. {'name': 'satori'}
  56. '''
  57. # 数字是12,所以指向第十二行,第二个YIELD_VALUE,yield的值就是456
  58. # 此时name="satori",被添加到了局部变量当中
  59.  
  60. # 因此到这里便更容易理解了,为什么生成器可以实现了。
  61. # 因为PyGenObject对函数的暂停和前进,进行了完美的监督,有变量保存我最近一行代码执行到什么位置
  62. # 再通过yield来暂停它,就实现了我们的生成器
  63.  
  64. # 跟函数一样,我们的生成器对象也是分配在堆内存当中的,可以像函数的栈帧一样,独立于调用者而存在
  65. # 我们可以在任何地方去调用它,只要我们拿到这个栈帧对象,就可以控制它继续往前走
  66. # 正是因为可以在任何地方控制它,才会有了协程这个概念,这是协程能够实现的理论基础
  67. # 因为有了f_lasti,生成器知道下次会在什么地方执行,不像函数,必须要一次性运行完毕
  68. # 以上就是生成器的运行原理

生成器栈帧特点

  生成器的栈帧与普通函数的栈帧并不相同

扩展:自定义序列实现迭代器

  1. # ==== 自定义序列实现迭代器 ====
  2.  
  3. class My_list(object):
  4. """为了能让该容器能够多次被循环调用,故做成可迭代对象。
  5. 每次for循环为其创建一个专属迭代器。"""
  6. def __init__(self,*args):
  7. self.args = args
  8. def __iter__(self):
  9. return My_list_iterator(self.args)

  10. class My_list_iterator(object):
  11. def __init__(self,args):
  12. self.args = args
  13. self.index = 0
  14. def __iter__(self):
  15. return self
  16. def __next__(self):
  17. try:
  18. return_value = self.args[self.index]
  19. except IndexError:
  20. raise StopIteration
  21. self.index += 1
  22. return return_value

  23. if __name__ == "__main__":
  24. # for循环原理。
  25. # 1.创建专属迭代器。
  26. # 2.不断执行next方法。
  27. # 3.捕捉StopIteration异常
  28. l = My_list(1,2,3,4,5)
  29. l_iterator = iter(l)
  30. while True:
  31. try:
  32. print(next(l_iterator))
  33. except StopIteration:
  34. break

自定义序列实现迭代器

  注意:即使没有__iter__方法。只要对象具有__getitem__也是可以间接的创建专属迭代器。但是效率偏慢。

扩展:如何区分索引取值与切片

  其实这一节本应该在Python基础全部学习完成后再放出来。但是这里实在忍不住,因为上面写了自定义序列。那么这里就想写一下这个。这一节与生成器并无任何关系。

  1. # === 如何区分索引取值与切片 ====
  2.  
  3. from numbers import Integral # 取值操作全部来源于该类

  4. class My_list(object):
  5. def __init__(self,args):
  6. self.args = args
  7. def __getitem__(self, item):
  8. cls = type(self) # <class '__main__.My_list'>
  9. if isinstance(item,Integral): #如果是取值操作
  10. print(item)
  11. return cls(self.args[item])#返回新的My_list实例,注意必须是可迭代形式
  12. elif isinstance(item,slice): #若是切片操作
  13. print(item) # slice(0, 2, None)
  14. return cls(self.args[item]) # 返回新的My_list实例
  15. def __repr__(self):
  16. return str(self.args)

  17. l = My_list([1,2,3])
  18. print(l[1])
  19. print(l[0:2])

如何区分索引取值与切片

扩展:range()方法的返回值与优化

  range()返回的是一个可迭代对象,但是range()这个可迭代对象并不像list那种可迭代对象一样真正占据内存空间。当for循环对其遍历的时候通过range()返回的可迭代对象本身__iter__方法创建出一个专属的迭代器。 然后其专属迭代器中的__next__方法里面是通过计算结束和步长的关系达到惰性求值的效果,range()__iter__方法创建出的迭代器并不属于生成器范畴但是有着和生成器异曲同工的作用。

  一句话总结:range()方法返回的可迭代对象并不存储具体的值,但是要对其进行遍历时创建的专属迭代器是具有惰性求值的特点的。 我将它称为优化后的可迭代对象,注意这个优化只是针对内存空间中的优化。但是它也有不方便的地方,就是不能通过index取值!!!

在此特别感谢武大神!!!

还有!我被武Sir夸了哈哈哈哈哈贴出来炫耀一下可以吹一年!

  1. # 另外附上群中大佬自己写的一个Range
  2. # 惰性求值,并且将本身做成了一个可重复调用的迭代器。
  3. # 总结:很强!!!
  4. class Range(object):
  5. def __init__(self,scale,s=0,d=1):
  6. self.s = s # 开始
  7. self.e = scale #总长度
  8. self.d = d # 步长

  9. def __iter__(self):
  10. self.i = self.s # 开始值
  11. return self

  12. def __next__(self):
  13. if self.i < self.e:
  14. x = self.i
  15. self.i += self.d
  16. return x
  17. else:
  18. self.i = self.s # self.i = 0
  19. raise StopIteration

  20. # 尽管Pyhon3中的range并不是直接返回一个迭代器本身。但是大佬的这种做法
  21. # 依然很厉害,__next__ 中的惰性求值也是和生成器有着异曲同工之妙。而且大佬本身自己就做成了一个可重复使用的迭代器。

Python三大器之生成器的更多相关文章

  1. python基础(补充):python三大器之生成器

    生成器的定义 通过列表生成式,我们可以直接创建一个列表.但是,受到内存限制,列表容量肯定是有限的.而且,创建一个包含100万个元素的列表,不仅占用很大的存储空间,如果我们仅仅需要访问前面几个元素,那后 ...

  2. Python三大器之迭代器

    Python三大器之迭代器 迭代器协议 迭代器协议规定:对象内部必须提供一个__next__方法,对其执行该方法要么返回迭代器中的下一项(可以暂时理解为下一个元素),要么就引起一个Stopiterat ...

  3. Python三大器之装饰器

    Python三大器之装饰器 开放封闭原则 一个良好的项目必定是遵守了开放封闭原则的,就比如一段好的Python代码必定是遵循PEP8规范一样.那么什么是开放封闭原则?具体表现在那些点? 开放封闭原则的 ...

  4. python三大器之while,if,for循环

    一.for循环(遍历循环) 在Python你可能要经常遍历列表的所有元素,对每个元素执行相同的操作;对于包含数字的列表,可能要对每个元素进行相同的计算;在网站中,可能需要显示文章中的每个标题等等.某一 ...

  5. python基础(补充):python三大器之装饰器

    函数作为返回值 高阶函数除了可以接受函数作为参数外,还可以把函数作为结果值返回. 我们来实现一个可变参数的求和.通常情况下,求和的函数是这样定义的: def calc_sum(*args): i = ...

  6. python三大器之装饰器的练习

    装饰器 加载顺序从下至上 执行顺序从上至下 ''' 多层装饰器 ''' def deco1(func): #func=deco2 def wrapper1(*args, **kwargs): '''t ...

  7. python三大神器之virtualenv pip, virtualenv, fabric通称为pythoner的三大神器。

    python三大神器之virtualenv   pip, virtualenv, fabric通称为pythoner的三大神器. virtualenv virtualenv------用来建立一个虚拟 ...

  8. python 迭代器和生成器

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

  9. 关于Python 迭代器和生成器 装饰器

    Python 简介Python 是一个高层次的结合了解释性.编译性.互动性和面向对象的脚本语言. Python 的设计具有很强的可读性,相比其他语言经常使用英文关键字,其他语言的一些标点符号,它具有比 ...

随机推荐

  1. 【MOOC操作系统】测试题大题-进程调度 先入先服务算法例题 【某多道程序系统供用户使用的主存为100K,磁带机2台,打印机1台,采用可变分区存储管理,静态方式分配外围设备(进程获得所需全部设备才能进入内容),忽略用户作业的I/O时间。采用动态分区、首次匹配法(从低地址区开始)分配主存,一个作业创建一个进程,且运行中不紧缩内存。作业调度采用FCFS算法,在主存中的进程采用剩余时间最短调度算法。】

    分析图: 答案: (1) 8 : 00作业1到达,占有资源并调入主存运行. 8: 20作业2和3同时到达,但作业2因分不到打印机,只能在后备队列等待.作业3资源满足,可进主存运行,并与作业1平分CPU ...

  2. 此flash player与您的地区不相容——更换新版本edge后出现的问题

    最新切换到了edge浏览器,使用flash时提示:"此flash player与您的地区不相容",而chrome是没有问题的.网上找到解决方案,发现一个可以有效解决的方式,如下: ...

  3. ms-setting是什么

    ms-settings 遇到了两个问题,记录一下 1)windows桌面右键菜单-->显示设置或者个性化-->报错:ms-settings:personalization-backgrou ...

  4. Java分层经验

    在学习和使用Java的过程中,我们时常要用到各种工具与技术,它们在某些时候可以大幅度地简化编程,利用好它们,可以让代码更强壮.下面的表格是我总结的关于java开发可能会用到的工具与它们在项目中扮演的角 ...

  5. Web-从Java Request对象到HTTP协议

    https://mp.weixin.qq.com/s/PjcA22STEDGwRxVQweObQQ Java Web中的Request对象是哪里来的?Response对象的角色是什么? Java We ...

  6. 使用Python代码制作GIF动态图

    使用Python  PIL.Image 制作GIF图片: import  PIL.Image                      相关模块 img = Image.open(img_name)  ...

  7. (Java实现)蓝桥杯Excel地址

    历届试题 Excel地址 原题地址 时间限制:1.0s 内存限制:256.0MB 提交此题 问题描述 Excel单元格的地址表示很有趣,它使用字母来表示列号. 比如, A表示第1列, B表示第2列, ...

  8. Javascript实现万年历(日历表)

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...

  9. java实现第六届蓝桥杯居民集会

    居民集会 蓝桥村的居民都生活在一条公路的边上,公路的长度为L,每户家庭的位置都用这户家庭到公路的起点的距离来计算,第i户家庭距起点的距离为di. 每年,蓝桥村都要举行一次集会.今年,由于村里的人口太多 ...

  10. java实现第八届蓝桥杯平方十位数

    平方十位数 题目描述 由0~9这10个数字不重复.不遗漏,可以组成很多10位数字. 这其中也有很多恰好是平方数(是某个数的平方). 比如:1026753849,就是其中最小的一个平方数. 请你找出其中 ...