匿名函数(lambda)在列表生成式和生成器中的应用示例
列表生成式中实例
先看题:
以下代码的输出是什么?请给出答案并解释:
def func(): return [lambda x: x * i for i in range(4)] print([j(2) for j in func()])
[6, 6, 6, 6]
解释:
函数 func() 中返回的是一个列表生成式,
走到这个列表生成式时,列表中生成的是一个个函数对象,且此时经过循环,i = 3;
Here We Go~~, 看下本质:
# 把上述函数中的返回值(列表表达式)赋值给 val val = [lambda x: x * i for i in range(4)] # 打印val print(val)
看下打印结果:
[<function <listcomp>.<lambda> at 0x000002970C18B158>, <function <listcomp>.<lambda> at 0x000002970C18B1E0>, <function <listcomp>.<lambda> at 0x000002970C18B268>, <function <listcomp>.<lambda> at 0x000002970C18B2F0>]
可以看到打印的结果是一个列表,里面放了4个函数对象;
也就是说,在执行 print([j(2) for j in func()]) 时, 执行了 func() 函数(内部执行__call__()),
func()函数返回值就是一个列表,里面放了4个函数对象;
简单说,此时 print([j(2) for j in func()]) ==> print([j(2) for j in "有四个函数对象的列表"])
for 循环这个列表, 循环一次就走一次 j(2),也就是走一次拿一个函数(函数名后面加括号即调用该函数);
所以最终输出 [6,6,6,6]
生成器中实例
变态后的需求
==> 请修改func的定义来产生期望的结果。
(也就是修改上面的个函数,实现出题人预期的结果…………无f**k可说)
def func(): return (lambda x: x * i for i in range(4)) print([j(2) for j in func()])
[0, 2, 4, 6]
对,两个函数的区别就是,一个是[ ]、一个是括号( );
列表时返回的是一个列表生成式,括号时返回的是???
解释:
看下打印结果:
由打印结果可以看到:
val的数据类型是一个“生成器”对象,也即一个可迭代对象;
所以,执行 return (lambda x: x * i for i in range(4)) 时,返回的直接是一个生成器对象(可迭代对象);
根据特性,不向生成器要东西,它就不给你生,即里面的 i 还是最初始的0;
但是,执行 return [lambda x: x * i for i in range(4)] 时,返回的是由列表生成式生成的一个列表;
根据特性,列表猴急的一下就把列表生成了,即里面的 i 直接是个定值了;