开局一张图总结关系

![](https://images2018.cnblogs.com/blog/1226829/201808/1226829-20180808170030116-52783735.png)

一、列表解析式

我们习惯生成列表通过list = [1, 2, 3]的方式。还有一种很方便的列表生成方式
list = [a*2 for a in range(10)],或者list = [fun(a) for a in range(10)]都是可以的

>>> L1 = [a*2 for a in range(10)]
>>> L1
[0, 2, 4, 6, 8, 10, 12, 14, 16, 18]

如果只是生成一个100元素,1000元素的列表,那很简单,可是你需要一个100万,一亿个元素的列表时,这样去生成列表,你要知道,它是实实在在的存在你的内存里面,这么大一个列表,过于耗费内存。

那么有没有什么好的解决办法呢?------------>生成器

二、生成器

按照某种算法推算出列表元素,可以一直一边循环一边计算出列表元素的机制,称之为生成器:generator。一个简单生成器构造:列表是L = [a*2 for a in range(10)] 生成器把中括号[]换成小括号()就好了,G = (a*2 for a in range(10)) 这样就得到了一个生成器,[.....]生成列表,(.....)生成生成器

>>> g = (a*a for a in range(10))
>>> g
<generator object <genexpr> at 0x0000001D925F3308> #1
>>> for v in g: #2
... print(v)
1
4
9
16
25
36
49
64
81
在#1处我们生成了一个generator obj ,通过#2 的方式就可以取到当中的元素了。
三、yield

首先我们来写一个斐波拉契数列 1, 1, 2, 3, 5, 8, 13, 21.....除第一个数外,每前两个数的和等于第三个数

fib_list = []
def fib(n):
i, a, b = 0, 0, 1
while i < n:
fib_list.append(b)
a, b = b, a+b
i = i + 1 f = fib(6)
print(fib_list) #结果为
[1, 1, 2, 3, 5, 8, 13, 21, 34, 55]

这样就得到了一个简单斐波拉契数列了。

当然我们还可以用yield来进行改装,把它变成一个生成器对象,yield: 在函数执行时会给函数返回generator对象,可以通过该对象的obj.next()方法或next(obj)来启动函数

>>> def generator_example():
... print('before yield')
... yield 'first yield' ... print('after first yield')
... message = yield 'second yield' #1 调用obj.send('message')可把值赋给yield表达式,同时返回yield右边的内容 ... print('第二个yield收到send函数的消息:{}'.format(message))
... yield
... return 'over'
>>> g = generator_example() #2 运行带yield的函数,返回一个generator obj
>>> g
<generator object generator_example at 0x0000006E725E3308>
>>> res = g.__next__() #3 调用obj.__next__()方法启动generator
before yield
>>> print(res)
first yield
>>> res2 = next(g) #4 next(obj) 等同于obj.__next__()
after first yield
>>> print(res2)
second yield
>>> res3 = g.send('form send: after second yield')
第二个yield收到send函数的消息:form send: after second yield
>>> print(res3)
None
>>> try:
... next(g)
... print('上一个next(g)因为没有yield,所以触发错误,这儿不会执行')
... except StopIteration as e:
... print(e.value)
over

上面没有提到obj.send()方法,#3处第一次通过obj.next()方法启动generator后,函数返回第一个yield右边的值,并停在了哪儿,然后第二次通过next(obj)让函数继续运行到第二个yield,返回second yield然后停住,然后第三次通过obj.send(value)的方法把value传给给了暂停地方的yield表达式的 message,并启动函数继续运行直到遇到下一个yield。......最后当触发最后次yield但是后面没有yield能暂停就报错了,使用try:抓住错误,并接受函数返回值over

yield来实现斐波拉契数列
def fib(n):
i, a, b = 0, 0, 1
while i < n:
yield b #在这儿用yield就好了
a, b = b, a+b
i = i + 1
return '--done--' f = fib(6)
while True:
try:
res = next(f)
print(res)
except StopIteration as e:
print(e.value)
break #运行结果为
1
1
2
3
5
--done--
四、迭代器

生成器都是迭代器

还有可以作用于for循环的数据类型有:

一类是集合数据类型:如 list、tuple、duct、str等

另一类是generator包括生成器,和带yield的generator function,这些可以直接作用于for循环的对象统称为可迭代对象(Iterable)可使用isinstance()来判断是否为Iterable

>>> from collection import Iterable
>>> isinstance([], Iterable)
True

而生成器不但可以作用于for循环,还能被next()函数不断调用返回下一个值,直到抛出StopIteration错误无法继续

可以被next()函数调用并不断返回下一个值的对象就称为迭代器Iterator

list、idct、str虽然可以用for循环,可迭代,但是没有nex()方法就不是Iterator迭代器

而使用iter()函数就可以把它们变成迭代器了

>>> gen_list = iter([1, 3, 5, 7])
>>> gen_list
<list_iterator object at 0x00000052371B6C50>

range() 或for line in f:这样的内部也都是封装了迭代器

python 生成器、列表解析式、yield、迭代器的更多相关文章

  1. [转]Python 的列表解析式,集合解析式,字典解析式

    Python 的列表解析式,集合解析式,字典解析式 这三种都是 python 里面的语法糖. 语法糖,Syntactic Sugar,就是为了写程序时候少出错,发明的一些简便的方法,但不影响这个语法的 ...

  2. python之列表推导、迭代器、生成器

    http://blog.chinaunix.net/uid-26722078-id-3484197.html 1.列表推导 看几个例子,一切就明白了. #!/usr/bin/python number ...

  3. Python 生成器的使用(yield)

    一. 生成器就是一个特殊的迭代器, 使用关键字yield就可以生成一个生成器 def func(): for i in range(10): yield i item = func() yield i ...

  4. Python之路(第十篇)迭代器协议、for循环机制、三元运算、列表解析式、生成器

    一.迭代器协议 a迭代的含义 迭代器即迭代的工具,那什么是迭代呢? #迭代是一个重复的过程,每次重复即一次迭代,并且每次迭代的结果都是下一次迭代的初始值 b为何要有迭代器? 对于序列类型:字符串.列表 ...

  5. Python之路迭代器协议、for循环机制、三元运算、列表解析式、生成器

    Python之路迭代器协议.for循环机制.三元运算.列表解析式.生成器 一.迭代器协议 a迭代的含义 迭代器即迭代的工具,那什么是迭代呢? #迭代是一个重复的过程,每次重复即一次迭代,并且每次迭代的 ...

  6. Python之列表生成式、生成器、可迭代对象与迭代器

    本节内容 语法糖的概念 列表生成式 生成器(Generator) 可迭代对象(Iterable) 迭代器(Iterator) Iterable.Iterator与Generator之间的关系 一.语法 ...

  7. python列表解析式,字典解析式,集合解析式和生成器

    一.列表解析式(列表推倒式): 功能:是提供一种方便的列表创建方法,所以,列表解析式返回的是一个列表. 1 lst = [1, 3, 5, 8, 10] 2 ll = [x+x for x in ls ...

  8. Python函数——列表推导式、生成器与迭代器

    列表推导式 产生背景 现在有个需求,看列表[0, 1, 2, 3, 4, 5, 6, 7, 8, 9],要求你把列表里的每个值加1,你怎么实现? 第一种方法: a = [1,3,4,6,7,7,8,9 ...

  9. 【转】Python之列表生成式、生成器、可迭代对象与迭代器

    [转]Python之列表生成式.生成器.可迭代对象与迭代器 本节内容 语法糖的概念 列表生成式 生成器(Generator) 可迭代对象(Iterable) 迭代器(Iterator) Iterabl ...

随机推荐

  1. POI生成Web版Word文件

    POI生成Web版Word文件 1       通过URL的输入流实现 2       直接把Html文本写入到Word文件 所谓的使用POI生成Web版Word文件是指利用POI将Html代码插入到 ...

  2. WPFの操作文件浏览框几种方式

    方式1: 使用win32控件OpenFileDialog Microsoft.Win32.OpenFileDialog ofd = new Microsoft.Win32.OpenFileDialog ...

  3. Django REST framework之版本,解释器,序列化

    1 版本 2 解释器 3.序列化 1 版本 通过?后面传版本号有两种方法: 方法一 from django.shortcuts import render from rest_framework.vi ...

  4. mini2440裸机试炼之—RTC闹钟中断,节拍中断

    版权声明:博客地址:http://blog.csdn.net/muyang_ren.源代码能够在我的github上找看看 https://blog.csdn.net/muyang_ren/articl ...

  5. SpringBoot实战(十四)之整合KafKa

    本人今天上午参考了不少博文,发现不少博文不是特别好,不是因为依赖冲突问题就是因为版本问题. 于是我结合相关的博文和案例,自己改写了下并参考了下,于是就有了这篇文章.希望能够给大家帮助,少走一些弯路. ...

  6. 1226 快速幂 取余运算 洛谷luogu

    还记得 前段时间学习二进制快速幂有多崩溃 当然这次方法略有不同 居然轻轻松松的 题目描述 输入b,p,k的值,求b^p mod k的值.其中b,p,k*k为长整型数. 输入输出格式 输入格式: 三个整 ...

  7. YOCVM

    一.热补丁的本质 对于线上紧急的bug,重新提审AppStore的时间过长.因此,能够下发一段补丁代码到线上运行,并结合Runtime,实时改变App原有的行为,就显得极为重要.补丁代码的形式可以有很 ...

  8. python简介及安装配置

    概述 python是解释型语言,相对编译型语言,执行效率较低.python是通过c语言编写,官方解释器也是c语言编写cpython,也有其他的如用java编写的jpython.目前有2.0和3.0版本 ...

  9. stream classdesc serialVersionUID = -7218828885279815404, local class serialVersionUID = 1.

    序列化类时出现的异常! 当某一个类实现java.io.Serializable接口时,该类默认会生成一个private static final long serialVersionUID = 1L; ...

  10. C. K-Dominant Character

    给出一个字母串,k满足:长度至少为k的字串一定包含某字母c,求最小的k 一个数组记录每个字母上一次出现的位置,用来计算另一个数组:记录每个字母与其相邻的相同字母的最大距离(设0和len两个位置一定有相 ...