Ⅰ起因

  学习python的同学通常会遇到这样一道经典生成器测试题:

def gen():
for i in range(4):
yield i
base = gen()
for n in (2,10):
base = (i+n for i in base)
print(list(base))
[21,22,23,24]

#简单解答:
因为for循环了两次,并对base从新赋值了,所以可以简化为(i+n for i in (i+n for i in base)) 而n 全部引用了后赋值的10。最里面的base引用的是gen。

答案及解释

   但是这个解答并没有回答一个核心问题:为什么最里层的n 始终用的是10,而base可以找到之前的gen()?

     为了简化问题,我把这道题改造了成这样:

a1 = 3
b = (i for i in range(a1))
a1 = 5
list(b) #[0, 1, 2] a1 = 3
b = (a1 for i in range(3))
a1= 5
list(b) #[5, 5, 5]

  或许各位会猜测:这个问题可能和for后面的数据类型有关系吧?

Ⅱ原理探索

  但如果把range()和前面的数值都改造为列表,结果如下:

a1 = 1
b=([a1,] for i in range(3))
a1=2
list(b) # [[2], [2], [2]] a1 =1
b = (i for i in [a1,])
a1 = 2
list(b) # [1] #也可以把以上两个表达式结合一下
a1 = 1
b = ([a1,i] for i in [a1,])
a1 = 2
list(b) #[[2, 1]]

显而易见,当变量在for前面的时候,会引用后声明的值,而当变量在for后面的iterator中的时候会引用之前声明的值,并且与数据类型无关。

By the way, 可能很人多还不确定列表本身的设定:a =1     b = [a,]       a =2     print(b)   #[1,]

当然以上全部是生成器表达式。如果手动定义一下生成器呢?

a =1
def zz():
for i in [a,]:
yield [a,i]
cc = zz()
a=2
print(list(cc)) #[[2, 2]] #如果传入a
a =1
def zz(a):
for i in [a,]:
yield [a,i]
cc = zz(a)
a=2
print(list(cc)) #[[1,1]]

  生成器函数的测试结果是前后一致,不存在这个问题。

  进一步测试:

a = 1
c = ([b,i] for i in [a,])
b = 1
list(c) #[[1, 1]] # 但是如果a在生成器表达式后面定义的话:
c = ([b,i] for i in [a,])
b = 1
a = 1
list(c) # 会报错 #p.s. 在生成器函数也不会报错

Ⅲ执行效率比较

  对于简单的生成器,生成器表达式更方便、更直观。那么两者的执行效率是否存在差异呢?Timeit!

import timeit
def b():
a = 9999
def c():
for i in range(a):
yield i
list(c())
print(timeit.timeit(stmt=b,number=1000))

函数模式

import timeit
def b():
a = 9999
c = (i for i in range(a))
list(c)
print(timeit.timeit(stmt=b,number=1000))

表达式

结果:

    函数模式    表达式模式

    1.260876    1.235369  
    1.253225    1.238639
    1.256804    1.235393
    1.258575    1.238165

我们看到生成器表达式提供的便利的确是以效率的损耗作为代价的。

进一步的验证表明:生成器表达式初始化的过程相比生成器函数需要花费更多的时间(接近2倍),但是由于初始化的时间过短,并不是形成差距的主要原因。函数模式的生成器会随着next()次数的增加在时间上逐步拉开与生成器表达式差距。调用效率的差距才是主要原因。

Ⅳ结论

  生成器表达式,会在程序执行的过程中运行for 后面的代码,并对for后面的代码进行赋值,而for之前的代码以及生成器函数并不会执行,只会进行编译。

  尽管,生成器表达式代码更简洁,但在生成器初始化和生成器调用的效率上都表现出了与传统生成器函数的差距。

注:列表推导式并不存在这样的问题(当然也不应该出现)

python 特别的生成器表达式的更多相关文章

  1. 05.python解析式与生成器表达式

    解析式和生成器表达式 列表解析式 列表解析式List Comprehension,也叫列表推导式 #生成一个列表,元素0-9,将每个元素加1后的平方值组成新的列表 x = [] for i in ra ...

  2. Python - 列表解析式/生成器表达式

    列表解析式: [expr for iter_var in iterable if cond_expr] 生成器表达式: (expr for iter_var in iterable if cond_e ...

  3. 详解Python中的生成器表达式(generator expression)

    介绍 1.生成器表达式(generator expression)也叫生成器推导式或生成器解析式,用法与列表推导式非常相似,在形式上生成器推导式使用圆括号(parentheses)作为定界符,而不是列 ...

  4. python基础之生成器表达式形式、面向过程编程、内置函数部分

    生成器表达式形式 直接上代码 1 # yield的表达式形式 2 def foo(): 3 print('starting') 4 while True: 5 x=yield #默认返回为空,实际上为 ...

  5. 详解python中的生成器表达式

    什么是生成器表达式 还记得列表解析吗?我们把[]换成()就变成生成器表达式了. g = (x for x in [1, 2, 3, 4]) print(g) # <generator objec ...

  6. 【Python】【容器 | 迭代对象 | 迭代器 | 生成器 | 生成器表达式 | 协程 | 期物 | 任务】

    Python 的 asyncio 类似于 C++ 的 Boost.Asio. 所谓「异步 IO」,就是你发起一个 IO 操作,却不用等它结束,你可以继续做其他事情,当它结束时,你会得到通知. Asyn ...

  7. Python 进阶_生成器 & 生成器表达式

    目录 目录 相关知识点 生成器 生成器 fab 的执行过程 生成器和迭代器的区别 生成器的优势 加强的生成器特性 生成器表达式 生成器表达式样例 小结 相关知识点 Python 进阶_迭代器 & ...

  8. python迭代器和生成器(3元运算,列表生成式,生成器表达式,生成器函数)

    1.1迭代器 什么是迭代器: 迭代器是一个可以记住遍历的位置对象 迭代器对象从集合的第一个元素元素开始访问,直到所有元素被访问完结束,迭代器只能往前不会后退. 迭代器有两个基本方法:iter ,nex ...

  9. Python学习笔记2:构造序列:列表推导和生成器表达式

    欢迎访问个人网站:www.comingnext.cn 1. 关于Python内置序列类型 a. 按能否存放不同类型的数据区分 容器序列: list.tuple 和collections.deque这些 ...

随机推荐

  1. 解决mysql设置时区时的错误Unknown or incorrect time zone: 'Asia/Shanghai'

    Mysql默认时区格式是'+8:00'的格式,这个时区可以在my.ini中[mysqld]节点下设置 default-time-zone = '+8:00' 默认这个设置是没有的 但是mysql不支持 ...

  2. 在Tomcat文件中,点击start.bat启动的是另一个tomcat

    遇到问题:在下载了一个免安装的Tomcat7之后解压,点击startup.bat启动tomcat,却报了异常. 后来在电脑一个文件夹中发现了另一个Tomcat8,是安装版本,并配置了环境变量.其环境变 ...

  3. mysql判断表里面一个逗号分隔的字符串是否包含单个字符串、查询结果用逗号分隔

    1.mysql判断表里面一个逗号分隔的字符串是否包含单个字符串 : FIND_IN_SET select * from tablename where FIND_IN_SET(传的参数,匹配字段) 例 ...

  4. 管理多个gradle,SDKMAN

    背景:同一台机器上有两个app需要编译,但是两个app的gradle版本不一致,所以需要安装一个管理gradle版本的工具   sdkman:(Software Development Kit Man ...

  5. selenium java 浏览器操作

    环境搭建 selenium 2.53 selenium-java-2.53.0.jar selenium-java-2.53.0-srcs.jar 原代码包 拷贝的工程lib下,做build path ...

  6. Json常用代码

    以下使用的都是fastJson. 先创建Person类,如下: public class Person { @JSONField(name = "AGE") private int ...

  7. windows server 2012R2 故障转移集群配置

    配置说明: AD:10.10.1.10/24  Node-2:10.10.1.20/24 Node-3:10.10.1.30/24 zhangsan-PC:10.10.1.50/24  VIP1:10 ...

  8. Codeforces Round #553 (Div. 2) A题

    题目网址:http://codeforces.com/contest/1151/problem/A 题目大意:给定一个由大写字母构成的字符串和它的长度,有这样的操作,使任意一个字母变成与其相邻的字母, ...

  9. Python数据库连接池DBUtils

    Python数据库连接池DBUtils   DBUtils是Python的一个用于实现数据库连接池的模块. 此连接池有两种连接模式: 模式一:为每个线程创建一个连接,线程即使调用了close方法,也不 ...

  10. try_files

    try_files $uri $uri/ /index.php$is_args$args 假设你防问 https://demo.com/demo 1.$uri:查找/demo文件 2.$ui/:查找/ ...