Python学习之Part22.匿名函数

匿名函数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()])
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值