Python匿名函数Lambda

一、概念介绍

1.匿名函数介绍

在Python中,通过lambda关键字来定义的函数称为匿名函数;
lambda函数能接收任何数量(可以是0个)的参数,但只能返回一个表达式的值,lambda函数是一个函数对象,直接赋值给一个变量,这个变量就成了一个函数对象
语法:lambda 参数 : 表达式
先写lambda关键字,然后依次写匿名函数的参数,多个参数中间用英文逗号分隔,然后是一个英文冒号,冒号后面写返回的表达式;

2.匿名函数与普通函数的对比

def sum_func(a, b, c):
    return a + b + c

sum_lambda = lambda a, b, c: a + b + c

print(sum_func(1, 100, 10000))
print(sum_lambda(1, 100, 10000))

运行结果:

10101
10101
  • 可以看到,lambda适用于多个参数、一个返回值的情况,可以用一个变量来接收,变量是一个函数对象,执行这个函数对象的结果与执行一个普通函数的结果一样。

  • lambda函数比普通函数更简洁,且没有声明函数名,上面的代码是用一个变量来接收lambda函数返回的函数对象,并不是lambda函数的名字。

3.匿名函数的多种形式

#无参数
lambda_a = lambda :1001
print(lambda_a())

#一个参数
lambda_b = lambda b : b * 10
print(lambda_b(10))

#多个参数
lambda_c = lambda a,b,c : a+b+c
print(lambda_c(5,6,7))

#表达式
lambda_d = lambda x: x if x % 2 ==0 else x+1
print(lambda_d(5))
print(lambda_d(6))

运行结果:

1001
100
18
6
6

lambda的参数可以是0个或多个,并且返回的表达式可以是一个复杂的表达式,只要最后返回的是一个值就可以了;

4.lambda作为一个参数传递

def sub_func(a,b,func):
    print('a = ',a)
    print('b = ',b)
    print('a + b = ',func(a,b))

sub_func(12,5,lambda a,b:a+b)

运行结果:

a =  12
b =  5
a + b =  17

5.lambda函数与python内置函数配合使用

list1 = [{'a': 10, 'b': 20}, {'a': 13, 'b': 2}, {'a': 23, 'b': 13}, {'a': 32, 'b': 17}]
max_a = max(list1, key=lambda x: x['a'])		# 列表元素中关键字'a'的最大值
max_b = max(list1, key=lambda x: x['b'])		# 列表元素中关键字'b'的最大值

print('列表最大数:', max_a)
print('列表最大数:', max_b)

运行结果:

列表最大数: {'a': 32, 'b': 17}
列表最大数: {'a': 10, 'b': 20}

6.lambda作为函数的返回值

def ret_func(a,b):
    return lambda c : a + b + c
return_func = ret_func(50, 51)
print(return_func)
print(return_func(4))

运行结果:

<function ret_func.<locals>.<lambda> at 0x000001711D2B8400>
105

匿名函数可以作为一个函数的返回值,在如上代码中,ret_func返回的是一个匿名函数,返回的是一个函数对象,当执行这个函数时,可以得到lamdba函数的结果。
注意:其中a,b两个参数是ret_func中的参数,但执行返回的函数return_func时,已经不在ret_func的作用域内,而lambda函数仍然能使用a,b,参数,说明lambda函数会将它的运行环境保存一份,一直保留到它自己执行的时候使用。

7.匿名函数的优点

  • 使用Python写一些脚本时,使用lambda可以省去定义函数的过程,让代码更精简;
  • 对于一些抽象的,不会被别的地方再重复使用的函数,有时候函数起个名称也是个难题,使用lambda不需要考虑命命名问题;
  • 使用lambda在某些时候代码更容易理解

二、使用场景

使用场景:
1.需要将一个函数对象作为参数来传递时,可以直接定义一个lambda函数(作为函数的参数或返回值);
2.要处理的业务符合lambda函数的情况(任意多个参数和一个返回值),并且只有一个地方会使用这个函数,不会在其他地方重用,可以使用lambad函数;
3.与一些python的内置函数配合合适,提高代码的可读性;

1、和map函数一起使用

map()函数接收两个参数,一个是函数,一个是Iterable,map将传入的函数依次作用到序列的每个元素,并把结果作为新的Iterator返回

遍历序列,对序列中每个元素进行函数操作,最终获取新的序列

案例如下:

求列表[1,2,3,4,5,6,7,8,9],返回一个n*n 的列表

#一般解决方案
li = [1,2,3,4,5,6,7,8,9]
for ind,val in enumerate(li):
    li[ind] = val * val
print(li)

[1, 4, 9, 16, 25, 36, 49, 64, 81]

$\textcolor{blue}{enumerate()是python的内置函数, 在字典赏识枚举、列举的意思;}$
$\textcolor{blue}{enumerate参数为可遍历/可迭代的对象(如列表、字符串);}$
$\textcolor{blue}{enumerate多用于在for循环中得到计数,利用它可以同时获得索引和值,即需要 index 和 value 值的时候可以使用enumerate;}$
$\textcolor{blue}{enumerate()返回的是一个enumerate对象 。}$

#高级解决方案
li = [1,2,3,4,5,6,7,8,9]
print(list(map(lambda x:x*x,li)))

[1, 4, 9, 16, 25, 36, 49, 64, 81]

2、与reduce函数一起使用

reduce把一个函数作用在一个序列[x1, x2, x3, ...]上,这个函数必须接收两个参数,reduce把结果继续和序列的下一个元素做累积计算,其效果就是:

reduce(func,[1,2,3]) 等同于 func(func(1,2),3),返回值为单个值

对于序列内所有元素进行累计操作

#接受一个list并利用reduce()求积
from functools import reduce
li = [1,2,3,4,5,6,7,8,9]
print(reduce(lambda x,y:x * y,li))

结果 : 123456789 = 362880

a = [12, 34, 56]

print reduce(lambda x , y: x + y, a)  # 102

print reduce(lambda x , y: x - y, a) # -78

3、和filter函数一起使用

filter()也接收一个函数和一个序列。和map()不同的是,filter()把传入的函数依次作用于每个元素,然后根据返回值是True还是False决定保留还是丢弃该元素。

对于序列中的元素进行筛选,最终获取符合条件的序列

在一个list中,删掉偶数,只保留奇数

li = [1, 2, 4, 5, 6, 9, 10, 15]
print(list(filter(lambda x:x % 2==1,li)))  # [1, 5, 9, 15]

回数是指从左向右读和从右向左读都是一样的数,例如12321,909。请利用filter()筛选出回数

li = list(range(1, 200))
print(list(filter(lambda x:int(str(x))==int(str(x)[::-1]),li)))

[1, 2, 3, 4, 5, 6, 7, 8, 9, 11, 22, 33, 44, 55, 66, 77, 88, 99, 101, 111, 121, 131, 141, 151, 161, 171, 181, 191]

4、和sorted函数一起使用

sorted(iterable, /, *, key=None, reverse=False)

接收一个key函数来实现对可迭代对象进行自定义的排序

可迭代对象:主要与列表,字符串,元祖,集合和字典

key:接受一个函数,根据此函数返回的结果,进行排序

reverse:排序方向,默认为从小到大,reverse=True为逆向

对列表按照绝对值进行排序

li= [-21, -12, 5, 9, 36]
print(sorted(li, key = lambda x:abs(x)))

[5, 9, -12, -21, 36]

"""
sorted()函数按照keys进行排序,并按照对应关系返回list相应的元素:

keys排序结果 => [5, 9, 12, 21, 36]
| | | | |
最终结果 => [5, 9, -12, -21, 36]
"""

把下面单词以首字母排序

li = ['bad', 'about', 'Zoo', 'Credit']
print(sorted(li, key = lambda x : x[0]))

输出['Credit', 'Zoo', 'about', 'bad']

"""
对字符串排序,是按照ASCII的大小比较的,由于'Z' < 'a',结果,大写字母Z会排在小写字母a的前面。
"""

假设我们用一组tuple表示学生名字和成绩:

L = [('Bob', 75), ('Adam', 92), ('Bart', 66), ('Lisa', 88)]

请用sorted()对上述列表分别按名字排序

print(sorted(L, key = lambda x : x[0]))

输出[('Adam', 92), ('Bart', 66), ('Bob', 75), ('Lisa', 88)]

再按成绩从高到低排序

print(sorted(L, key = lambda x : x[1], reverse=True))

输出[('Adam', 92), ('Lisa', 88), ('Bob', 75), ('Bart', 66)]

5、与三元运算结合

如下:

#if 条件为真的时候返回if前面内容,否则返回0 
exp1= lambda x:x+1 if 2==1 else 0 
print(exp1(2))
exp2 = lambda x:x+1 if 1==1 else 0 
print(exp2(2))

输出:

0
3

#if not 为假返回if not前面内容,否则返回0  
exp3 = lambda x:x+1 if not 2==1 else 0  
print(exp3(2))  

exp4 = lambda x:x+1 if not 1==1 else 0  
print(exp4(2))

输出:
3
0