匿名函数lambda
首先我们很容易写出一个普通的函数,如下:
def calc(n):
return n**2
print(calc(10))
将此函数等价的换成匿名函数如下:
calc = lambda n:n**2
print(calc(10))
其中 calc 是函数名, lambda 定义匿名函数的关键字
以冒号为分隔,冒号之前的是参数,冒号之后的是表达;
匿名函数说明:
1.格式: 函数名 = lambda 参数:表达式
2.参数可以有多个,用逗号隔开
3.匿名函数不管逻辑有多复杂,只能写一行,且逻辑执行结束后的内容就是返回值
注意: lambda函数大概率会与 min,max,filter,map,reduce 这些重要的内置函数合用
匿名函数的用法有2种:
1.将函数赋值再掉用,例如:
add = lambda x,y:x+y
print(add(12.13,12.89))
2.不赋值,与其他函数合作,这也是匿名函数优势
例如:求出字典中value值最大对应的key值
dic = {'k1':10, 'k2':100, 'k3':30}
print(max(dic, key=lambda k:dic[k]))
匿名函数所需要注意的地方:
lambda 表达式本身不会返回值,它返回的是一个函数指针,就是一个函数在内存中的地址,因此,我们要得到函数具体的返回值,需要对返回的函数进行调用才行。
用一个例子进行说明:
题目:现有2个元组,((‘a’),(‘b’)),((‘c’),(‘d’)),使用匿名函数生成列表[{‘a’:‘c’},{‘b’:‘d’}]
print([lambda i:{i[0]:i[1]} for i in zip((('a'), ('b')),(('c'), ('d')))])
如果我们使用上面的代码,会发现输出的结果是这样的:
输出的是2个内存地址而不是我们想得到的值,是两个函数指针,我们如想得到函数执行的结果,需要对函数进行调用才行,有下面2种方式来实现:
1.赋值再调用:
fun = lambda i:{i[0]:i[1]}
print([fun(i) for i in zip((('a'), ('b')),(('c'), ('d')))])
2.直接调用匿名函数:注意要对 lambda整体加括号
print([(lambda i:{i[0]:i[1]})(i) for i in zip((('a'), ('b')),(('c'), ('d')))])
3.对于本题,还可以与其他高阶函数合用来解决:
ret = zip((('a'), ('b')),(('c'), ('d')))
res = map(lambda t:{t[0]:t[1]},ret)
print(list(res))
2道题分享:
以下代码的输出是什么,请给出答案并解释:
def multipliers():
return [lambda x:x*i for i in range(4)]
print([m(2) for m in multipliers()])
由于函数 multipliers 的返回是一个列表,在返回时,for循环中的 i 传参到匿名函数中时并没有被计算,因为此时的 x 值还没有被传递,所以旧的 i 值会被新的 i 覆盖,但是匿名函数一直未执行只有等到调用时才被执行,因此,当 for 循环结束时的最后一个 i 值才会被保留,成为一个固定的值,即返回的4个函数中 i = 3 ,当调用函数时: m(2), 即 x =2,则输出结果为 [6,6,6,6].
上面函数的与以下函数等价:
def multipliers():
li = []
for i in range(4):
li.append(lambda x:x*i)
return li
print([m(2) for m in multipliers()])
修改multipliers的定义产生期望的结果:
经过以上分析吗,我们知道输出结果有问题的原因是循环变量 i 在循环的过程中一直没有使用,循环结束才被使用,那么我们使每循环一次就执行一次函数即可,因此可以修改如下:
def multipliers():
# 修改为生成器表达式即可(它的惰性计算)
return (lambda x:x*i for i in range(4))
print([m(3) for m in multipliers()])