我们先考虑一个场景:

  有个情景需要循环输出1——10.

  这里给两种方法:

list1 = [1,2,3,4,5,6,7,8,9,10]
for i in list1:
print(i)
for i in range(1,11):
print(i)

两种方式输出结果一样,但是我们考虑一下,如果要求输出1——1000000呢?

第一种方式会导致list1里面真实放入1000000长度的数字,占用空间很大,明显不是明智之举,

再来看第二种方法,用到range帮助我们生成数据,在python3中range的本质就是一个生成器。

在python2中:range返回的是一个等差列表,比如[0,1,2,3,4,5,6,7,```````], 而xrange才是返回一个生成器对象. 即python2 range()==[```````````````````], python2 xrange()==python3 range()

具体对比查看:https://blog.csdn.net/humanking7/article/details/45950967

(一)这里写一个函数,在生成器函数的名称中加上gen 前缀或后缀,不过这不是必要的习惯:

 def gen_create_range(start,end):

     while start < end:
yield start
start += 1 for i in gen_create_range(1,5):
print(i)

 #output:

1

    2

    3

    4

这个函数没有return 但是可以有返回值,注意看里面有个yield关键字,这个函数和range()函数很像。

(二)什么是生成器:函数定义中包含yield关键字那么函数就变成了生成器

  概念:如果列表元素可以按照某种算法推算出来,那我们是否可以在循环的过程中不断推算出后续的元素呢?这样就不必创建完整的list,从而节省大量的空间。在Python中,这种一边循环一边计算的机制,称为生成器:generator   

  生成器表达式: 通列表解析语法,只不过把列表解析的[]换成()  

g= (x**2 for x in range(5))

print g

>>
<generator object <genexpr> at 0x0000000002771798> #如果、
L=[x**2 for x in range(5)] print L >>
[0, 1, 4, 9, 16]

  也就是说:创建L和g的区别仅在于最外层的[]和(),L是一个list,而g是一个generator

1).函数中只要出现了yield语句就会将其转变成一个生成器函数

  • 特别之处在于,yield 的作用就是把一个函数变成一个 generator,带有 yield 的函数不再是一个普通函数,Python 解释器会将其视为一个 generator

  • 与普通函数不一样,生成器值会在迭代操作的时候才能运行.yiled可以把函数中断,保存状态和继续执行的能力

好比一个武打片里面的慢镜头回放,yield把函数里面你要保存的值中断并保存,你通过调用next()来回放

 

比如:

def countdown(n):
print('Starting to count from',n)
while n>0:
yield n
n-=1
print('done') c=countdown(3)
print(c)
>>
<generator object countdown at 0x0000000002821828>
#表示这是一个生成器

2).调用该generator时,首先要生成一个generator对象,然后用next()函数不断获得下一个返回值

比如:

c=countdown(3)

#run the first yield and emit a value

print(next(c))

>>

Starting to count from 3

3

#run the next yield

print(next(c))

>>

2

#run the next yield

print(next(c))

>>

1

#run the next yield

print(next(c))

>>

done

    print next(c)

StopIteration

深入解释:

你把yield想象成时间断点,运行一次next就回放一下,看起来就好像一个函数在正常执行的过程中被 yield 中断了数次,每次中断都会通过 yield 返回当前的迭代值

  • 第一次next()是打印了 print('Starting to count from',n),提取第一次的保存值是3

  • 第二次再运行next()是继续在while里面的断点接着走,所以没有打印print('Starting to count from',n), 而是直接提取第二次的保存值2

  • 第三次再运行next()是继续在while里面的断点接着走,所以直接输出1

  • 第四次再运行next()的时候,发现yield缓存的武打片慢镜头都已经放完了,所以输出done之后,报了个错StopIteration

3).生成器函数用for循环

for n in countdown(6):
  
  print(n) >> Starting to count from 6 6 5 4 3 2 1 done

 

正确的方法是使用for循环,因为generator也是可迭代对象

4yield 与 return相爱相杀

1).在一个生成器中,如果没有return,则默认执行到函数完毕时返回StopIteration

def gen1():
yield 100 g1=gen1()
print(next(g1))
>>100

第一次调用next(g1)时,会在执行完yield语句后挂起,所以此时程序并没有执行结束。

print(next(g1))
>>
Traceback (most recent call last):
File "C:/about_gen.py", line 71, in <module>
print(next(g1))
StopIteration

程序试图从yield语句的下一条语句开始执行,发现已经到了结尾,所以抛出StopIteration异常

 

2).如果遇到return,如果在执行过程中 return,则直接抛出 StopIteration 终止迭代

def gen2():
yield 200
return
yield 300 g2=gen2()
print(next(g2))
>>200
# 程序停留在执行完yield 200语句后的位置 print(next(g2))
>>
File "C:/about_gen.py", line 82, in <module>
print(next(g2))
StopIteration

程序发现下一条语句是return,所以抛出StopIteration异常,这样yield 'b'语句永远也不会执行

生成器这个概念一开始很难理解,有点古怪,但是时间久了才知道他的妙用

另外生成器函数是没有办法使用return来返回值

python中的生成器(一)的更多相关文章

  1. python中的生成器函数是如何工作的?

    以下内容基于python3.4 1. python中的普通函数是怎么运行的? 当一个python函数在执行时,它会在相应的python栈帧上运行,栈帧表示程序运行时函数调用栈中的某一帧.想要获得某个函 ...

  2. Python学习-39.Python中的生成器

    先回顾列表解释 lista = range(10) listb = [elem * elem for elem in lista] 那么listb就将会是0至9的二次方. 现在有这么一个需求,需要存储 ...

  3. python中的生成器(二)

    一. 剖析一下生成器对象 先看一个简单的例子,我们创建一个生成器函数,然后生成一个生成器对象 def gen(): print('start ..') for i in range(3): yield ...

  4. Python中的生成器与yield

    对于python中的yield有些疑惑,然后在StackOverflow上看到了一篇回答,所以搬运过来了,英文好的直接看原文吧. 可迭代对象 当你创建一个列表的时候,你可以一个接一个地读取其中的项.一 ...

  5. 聊聊Python中的生成器和迭代器

    Python中有两个重要的概念,生成器和迭代器,这里详细记录一下. 1. 生成器 什么是生成器呢? 通过列表生成式,我们可以直接创建一个列表.但是,受到内存限制,列表容量肯定是有限的.而且,创建一个包 ...

  6. python中的生成器(generator)总结

    1.实现generator的两种方式 python中的generator保存的是算法,真正需要计算出值的时候才会去往下计算出值.它是一种惰性计算(lazy evaluation). 要创建一个gene ...

  7. python学习之【第十三篇】:Python中的生成器

    1.为什么要有生成器? 在Python中,通过列表生成式,我们可以直接创建一个列表.但是,受到内存限制,列表容量肯定是有限的.而且,创建一个包含100万个元素的列表,不仅占用很大的存储空间,如果我们仅 ...

  8. python中的生成器、迭代器、闭包、装饰器

    迭代器 迭代是访问集合元素的一种方式.迭代器是一个可以记住遍历的位置的对象.迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束.迭代器只能往前不会后退. 可迭代对象 以直接作用于 for ...

  9. python中的生成器和迭代器

    前言: 我们来了解一下什么是python中生成器.了解一下python生成器是什么,以及生成器在python编程之中能起到什么样的作用. 定义: 生成器和迭代器 通过列表生成式,我们可以直接创建一个列 ...

随机推荐

  1. Android-SDCardUtil-工具类

    SDCardUtil-工具类,是专门处理,外置存储Sdcard的操作 package common.library.utils; import android.annotation.SuppressL ...

  2. php萌新|学习|排坑|のmysqli_error()方法的妙用

    从开始学习php当现在已经有一个月多.除了每天完成公司布置的日常汇报,也没有耐下性子写一写自己想写的东西.今天就当起个头,坚持一周有个两三片文章或者小总结,也不枉费自己的付出.(我自己都不信,你会信吗 ...

  3. django drf 权限permission

    https://www.django-rest-framework.org/api-guide/permissions/#custom-permissions from django.shortcut ...

  4. iOS 获取 UITabViewController 和 UINavigationController 的图标位置

    这些图标是放在 UITabBar 和 UINavigationBar 里的.所以只要遍历它们的 subViews,找到类型是 UIButton 的就可以了. 所有想获取它们的相对位置很容易. 获取到相 ...

  5. GO学习笔记 - 函数名前面是否有输入参数肯定是不一样的!!

    在刚接触GO语言时候,我相信你也会有这种困惑,为什么有的函数名前面有输入参数,而一些却没有,它们是否有差别?确实有差别,没有输入参数,是一般的函数:有输入参数,是结构的方法,输入参数叫做“方法接收者” ...

  6. 七、linux目录结构知识---实战

    1.企业面试题:一个100M的磁盘分区,分别写入1k文件,及写入1M的文件,分别可以写多少个? 一块磁盘被分区格式化成系统文件后,有Inode和Block:一个文件一般占用一个Inode和一个Bloc ...

  7. 洛谷P1393 动态逆序对(CDQ分治)

    传送门 题解 听别人说这是洛谷用户的双倍经验啊……然而根本没有感觉到……因为另外的那题我是用树状数组套主席树做的……而且莫名其妙感觉那种方法思路更清晰(虽然码量稍稍大了那么一点点)……感谢Candy大 ...

  8. eclipse打包jar文件

    论文仿真做线性回归分类在人脸识别中应用与研究,在单机下实现LRC算法后,又在Hadoop云平台下实现了该算法.在比较实验结果时候需要放在相同硬件条件下比较.但是LRC单机算法是在windows下的ec ...

  9. [CSS3] 边栏导航动画

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

  10. jquery源码解析:jQuery数据缓存机制详解2

    上一课主要讲了jQuery中的缓存机制Data构造方法的源码解析,这一课主要讲jQuery是如何利用Data对象实现有关缓存机制的静态方法和实例方法的.我们接下来,来看这几个静态方法和实例方法的源码解 ...