[oldboy-django][5python基础][高级特性]generator生成器
# 生成器基础
- 定义
在循环的时候不断推算下一个元素的值,而不是一下子创建空间存储所有元素,这样节省空间。
并且在适当的条件结束循环,这种一边循环一边计算的机制,称为generator生成器 - 生成器创建方法(两种)
a.将列表生成式的[]改成()-- 称为生成器表达式
- 列表表达式 l = [x**x for x in range()]
- 生成器: g = (x**x for x in range())
b.带yield的函数
-- 当列表生成表达式比较复杂的时候,可以用函数来实现
ps:
range() 和xrange()区别
range()是一个列表,而xrange()是一个生成器
print(range)得到是[,,,,,,,,]
print(xrange)得到的一个生成器对象 range()会在内存中创建10个数字,而xrange不会创建,只有在循环的时候才会创建每个数字 - yield生成器(我简单的将带yield的函数称为yield生成器)
a.自定义一个生成器
def nrange(num):
temp = -
while True:
temp = temp +
if temp >= num:
return
else:
yield temp b. 使用生成器方法(两种)
- next
程序:
g = nrange()
print(next(g))
print(next(g))
print(next(g))
print(next(g))
print(next(g))
print(next(g)) 程序输出:
Traceback (most recent call last):
File "F:/oldboy/生成器/自定义生成器.py", line , in <module>
print(next(g))
StopIteration 可以看到每次调用next(g),就计算出g的下一个元素的值,
直到计算到最后一个元素,没有更多的元素时,抛出StopIteration的错误。
这种方法不是很好使,而且还没有对异常StopIteration处理。 - for循环
g = nrange()
for num in g:
print(num) 通过for循环来迭代它,并且不需要关心StopIteration的错误。 - yield生成器生命周期
也就是什么时候结束for循环,对函数改造的生成器,当执行到return或者函数最后一条语句时
就结束for循环 - yield生成器和函数不同
如果一个函数定义中包含yield关键字,那么这个函数就不再是一个普通函数,而是一个generator;
区别在于:执行流程不一样。
函数是顺序执行,遇到return就返回;而生成器在每次调用next的时候,遇到yield返回,
下次调用next的时候,接着上次的yield次继续执行 # 生成器使用next和send区别
send:
.使用send前必须使用了一次next
.执行g.send()时,会将send函数参数即10视为yield表达式(注意不是yield右侧的表达式)的结果,
然后,程序会继续推进到下一个yield那里,
最后,将yield 右侧的表达式(注意不是yield表达式)的结果作为send()函数的返回值,返回给外界。 next:
.第一次使用Next时,遇到yield返回,并将yield右边表达式的结果作为next的值返回
.之后使用next和send基本一样,不同的是将None作为yield表达式的结果,
然后,程序会继续推进到下一个yield那里,
最后,将yield右侧的表达式结果作为next()函数的返回值,返回给外界 # m = yield value 理解
只要理解一句话m = yield ,是将表达式"yield 5" 的结果返回给m, 而不是5,
而yield 5表达式结果的值和send, next有关。
如果是send(arg)是将arg作为 yield 5表达式结果赋给m
如果是next,是将None作为yield 表达式结果赋给m;
然后程序推进到下一个yield那里
最后,将yield右边(注意右边两字)的表达式“”作为next或者send函数的返回值,返回给外界 # 实例next
# -*- coding: utf- -*-
def f():
print("start")
current = yield "hello"
print('current=', current)
while True:
value = yield "bad"
print("value=",value)
# value = value + 'not' # 此行会报错,因为value会为None, 不能和字符串进行相加 g = f()
s1 = next(g)
print('s1=',s1)
# 第一次Next时,停止在第5行, 将yield右边的表达式(无即None)作为next()的返回值 s2 = next(g)
print('s2=', s2)
# 第二次的next,会将None作为yield表达式" yield hello"的值赋给current, 即current = None
# 然后程序往下执行,遇到yield "bad"停止,将yield右边的表达式"bad"作为第二次next的返回值 s3 = next(g)
print('s3=',s3)
# 第三次next,会将None作为yield表达式”yield bad"的值赋给value, 即value = None,
# 程序往下执行,遇到value = yield "bad"停止,将"bad"作为第三次next的返回值 # 实例send
# -*- coding: utf- -*-
def f():
print("start")
current = yield "hello"
print('current=', current)
while True:
value = yield "bad"
print("value=",value)
# value = value + 'not' # 此行会报错,因为value会为整形, 不能和字符串进行相加
g = f()
s1 = next(g)# 这一步不能少
print(s1)
# 第一次Next时,停止在第5行, 将yield右边的表达式(无即None)作为next()的返回值 s2 =g.send()
print(s2)
# 生成器调用第一次send时,将send参数10代替表达式"yield hello"的值,赋给current,
# 然后程序往下执行,在遇到yield bad停止,将"bad"作为第一次send()函数的返回值返给外界 s3 = g.send()
print(s3)
# 生成器第二次调用send时,将send参数20代替表达式"yield bad"赋给 value
# 程序往下执行,然后遇到yield bad停止,将"bad"作为第二次send()函数的返回值返给外界
[oldboy-django][5python基础][高级特性]generator生成器的更多相关文章
- [oldboy-django][5python基础][高级特性]Iterator迭代器
# 区分可迭代对象iterable, 迭代器iterator, 生成器generator a. iterable 可直接用for循环的对象,都称为可迭代对象, from collections imp ...
- python基础——高级特性
1.切片 切片: >>> L = ['Michael', 'Sarah', 'Tracy', 'Bob', 'Jack'] >>> L[:3] ['Michael ...
- Python学习札记(十八) 高级特性4 生成器
参考:生成器 Note 1.通过列表生成式,我们可以直接创建一个列表.但是,受到内存限制,列表容量肯定是有限的,且容易造成空间浪费.所以,如果列表元素可以按照某种算法推算出来,那我们可以在循环的过程中 ...
- [python学习篇][廖雪峰][1]高级特性--创建生成器 方法1 a = (x for x in range(1,3))
创建一个生成器的方法: for x in range(1,10000000) ,先生成一个列表[1........9999999] 如果我们只想要后面的几个元素,会发现浪费很多空间.所以,如果列表元素 ...
- [python学习篇][廖雪峰][1]高级特性--创建生成器 方法2 yield
def fib(max): n, a, b = 0, 0, 1 while n < max: print b a, b = b, a + b n = n + 1 将print b 改成yield ...
- 三、python高级特性(切片、迭代、列表生成器、生成器)
1.python高级特性 1.1切片 list列表 L=['Mli','add','sal','saoo','Lkkl'] L[0:3] #即为['Mli','add','sal'] 从索引0开始 ...
- Python高级特性(切片,迭代,列表生成式,生成器,迭代器)
掌握了Python的数据类型.语句和函数,基本上就可以编写出很多有用的程序了. 比如构造一个1, 3, 5, 7, ..., 99的列表,可以通过循环实现: L = [] n = 1 while n ...
- python高级特性:切片/迭代/列表生成式/生成器
廖雪峰老师的教程上学来的,地址:python高级特性 下面以几个具体示例演示用法: 一.切片 1.1 利用切片实现trim def trim(s): while s[:1] == " &qu ...
- python的高级特性:切片,迭代,列表生成式,生成器,迭代器
python的高级特性:切片,迭代,列表生成式,生成器,迭代器 #演示切片 k="abcdefghijklmnopqrstuvwxyz" #取前5个元素 k[0:5] k[:5] ...
随机推荐
- 在vue-cli中引入图片不能正常显示
我们用vue-cli构建项目的时候,图片的地址是后台的,可是在template中item.img放到src中是不能正常显示的为什么? 原因是:url-loader无法解析js动态生成的路径. 解决: ...
- 初见微服务之RESTful API
1. REST名称由来 REST全称为Representational State Transfer,即表述性状态转移,最早由Roy Feilding博士在世纪之交(2000年)提出,喜欢追根溯源的朋 ...
- IOS UIActivityIndicatorView动画
● 是一个旋转进度轮,可以用来告知用户有一个操作正在进行中,一般 用initWithActivityIndicatorStyle初始化 ● 方法解析: ● - (void)startAnimating ...
- IOS @param view 需要获取层级结构的view
- (void)applicationDidBecomeActive:(UIApplication *)application { NSString *str = [self digView:self ...
- Linux系统运维常见面试简答题(36题)
1.请描述下linux 系统的开机启动过程开机加电BIOS自检———–>MBR引导———–>grub引导菜单———–>加载内核———–>启动init进程———–>读取in ...
- cd ..和cd -
cd ..是返回上一层目录, cd -是返回到上一次的工作目录.
- 前端异常日志监控 - 使用Sentry
背景 现在的前端项目越来越复杂,在不同的客户端会产生各种在开发人员机器上不会出现的问题.当用户报告一个问题给开发人员的时候,开发人员无法直接定位问题.在此前,听过一次鹅厂的前端人员,他们对QQ里面的网 ...
- Apache RocketMQ 正式开源分布式事务消息
近日,Apache RocketMQ 社区正式发布4.3版本.此次发布不仅包括提升性能,减少内存使用等原有特性增强,还修复了部分社区提出的若干问题,更重要的是该版本开源了社区最为关心的分布式事务消息, ...
- cocoapods 类库管理利器
作为iOS开发者,第三方类库的使用是最经常的,但鉴于第三方类库的不断更新以及其可能需要依存其他类,如果要使用最新版那么我们需要重新下载再添加到项目中,无疑带来一些繁琐的麻烦,那么现在这里就有一款能解决 ...
- Mysql--子查询、分页查询、联合查询
一. 子查询的定义 出现在其他语句中的select语句,称为子查询或者内查询,外部的查询语句称为主查询或者外查询,子查询可以包含普通select可以包含的任何语句. 外部查询:select.inser ...