写python的时候,大多数场景下,我都是if else选手,因为最核心的逻辑几乎都是通过if else语句来实现的。关于匿名函数这块儿,其实可以用常见的循环等方法来实现,但是如果你想成为一个python的高手,匿名函数还是必须要了解的。因为匿名函数,能够让你的代码足够简洁,
01 什么是匿名函数?
在python中,匿名函数,顾名思义,就是没有名字的函数,它主要用在那些只使用一次的场景中。如果我们的程序中只需要调用一次某个简单逻辑,把它写成函数还需要先定义、取函数名字等一些列操作,这种场景下使用匿名函数往往能够让你的程序更加简单。
匿名函数还有名称,叫做lambda。匿名函数格式如下:
lambda arg1,arg2 ...,argN : expression
它常用格式是lambda关键字+逗号分隔的参数+冒号+表达式。
简单看个例子吧:
1
2
3
4
5
6
7
8
9
10
|
- - - - 计算一个数的平方 - - - >>> lambda x: x * * 2 <function < lambda > at 0x7f6ebe013a28 > - - - 注意,这个是一个函数的地址 - - - >>> func = lambda x: x * * 2 >>> func( 2 ) 4 >>> >>> func( 3 ) 9 |
利用lambda,我们实现对一个数字x求平方的运算,在python中,**代表乘方操作。
上面的例子中,x就是参数,冒号后面的x**2就是expression表达式。
当然,我们也可以定义一个函数来实现乘方操作。
lambda区别于函数的一点在于,lambda是一个表达式,它不是一个函数,也不是一个语句。因此,lambda可以被用在一些特殊的地方,例如下面的场景:
我们可以用range函数来生成一个list,如下:
1
2
3
|
>>> a = [ range ( 10 )] >>> a [[ 0 , 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 ]] |
如果我们要对这些数字做乘方操作,则可以直接写成下面的样子:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
>>> b = [( lambda x: x * x)(x) for x in range ( 10 )] >>> b [ 0 , 1 , 4 , 9 , 16 , 25 , 36 , 49 , 64 , 81 ] - - - - 如果用函数来实现,会发现报错 - - - >>> def fun(x): ... return x * * 2 ... >>> >>> c = [fun( range ( 10 ))] Traceback (most recent call last): File "<stdin>" , line 1 , in <module> File "<stdin>" , line 2 , in fun TypeError: unsupported operand type (s) for * * or pow (): 'list' and 'int' |
当然,你也可以利用函数,使用另外的方法来实现这个过程如下:
1
2
3
4
5
6
7
8
|
>>> def fun2(x): ... return x * * 2 ... >>> c = [] >>> for i in range ( 10 ): ... c.append(fun2(i)) >>> c [ 0 , 1 , 4 , 9 , 16 , 25 , 36 , 49 , 64 , 81 ] |
相比于前面的lambda函数,这个方法显得不那么灵巧。
02 函数式编程
所谓的函数式编程,代表代码中每块都是不可变的,都是由函数构成的,函数本身相互独立,互不影响,相同的输入都对应相同的输出,函数式编程特性,和lambda有密切的关系。
来看下面的例子,假如我们想让一个list中的元素都乘以2,可以写成下面的形式:
1
2
3
4
5
6
7
8
9
10
|
>>> l = [ 1 , 2 , 3 , 4 , 5 ] >>> def double_num(l): ... for index in range ( 0 , len (l)): ... l[index] * = 2 ... return l ... >>> double_num(l) [ 2 , 4 , 6 , 8 , 10 ] >>> l [ 2 , 4 , 6 , 8 , 10 ] |
上面这段代码,就不是一个函数式编程的例子。
因为每次输入列表L,L的值都会被改变,如果我们多次调用double_num这个函数,每次的结果都不一样。
那么如果我们让它变成一个函数式编程,就得写成下面这样:
1
2
3
4
5
6
7
8
9
10
11
|
>>> l = [ 1 , 2 , 3 , 4 , 5 ] >>> def double_num1(l): ... new_list = [] ... for index in l: ... new_list.append(index * 2 ) ... return new_list ... >>> double_num1(l) [ 2 , 4 , 6 , 8 , 10 ] >>> l [ 1 , 2 , 3 , 4 , 5 ] |
在python中,提供了常用的几个函数map、filter、reduce同lambda一起使用,来实现函数式编程(注意,这3个函数需要在python3的环境下使用)。
map函数 map(function, list)
注意,这里的function可以是匿名函数,也可以是普通的函数。
还是上面的乘以2的例子,假如我们使用map函数配合lambda来实现,可以写成下面这样:
1
2
3
4
5
6
7
8
9
10
|
>>> l = [ 1 , 2 , 3 , 4 , 5 ] >>> new_list = map ( lambda x: x * 2 , l) >>> for i in new_list: ... print (i) ... 2 4 6 8 10 |
这里的lambda就可以用函数来替换,如下:
1
2
3
4
5
6
7
8
9
10
11
12
|
>>> l = [ 1 , 2 , 3 , 4 , 5 ] >>> def double_x(x): ... return x * 2 >>> res = map (double_x, l) >>> for i in res: ... print (i) ... 2 4 6 8 10 |
filter函数 filter(function,list)
filter函数主要用来对可迭代的对象中的每个元素,都用function判断,将返回true的对象返回,返回false的对象抛弃,如下为判断一个集合中的偶数:
1
2
3
4
5
6
7
|
>>> l = [ 1 , 2 , 3 , 4 , 5 ] >>> new_l = filter ( lambda x: x % 2 = = 0 , l) >>> for i in new_l: ... print (i) ... 2 4 |
reduce函数 reduce(function, list)
reduce主要用来对一个列表做一些累计操作,假如我们要计算某个列表的累计乘积,可以用下面的方法:
1
2
3
4
5
|
>>> from functools import reduce >>> l = [ 1 , 2 , 3 , 4 , 5 ] >>> product = reduce ( lambda x,y: x * y, l) >>> product 120 |
03 lambda的性能如何?
下面是一个例子,测试不同的方案下,使用lambda、for循环和新建list的方法,分别对一个集合元素乘以2,计算的耗时情况:
1
2
3
4
5
6
7
8
|
[root@VM - 0 - 14 - centos ~] # python3 -mtimeit -s'a=range(1000)' 'map(lambda x: x*2, a)' 1000000 loops, best of 3 : 0.538 usec per loop [root@VM - 0 - 14 - centos ~] # python3 -mtimeit -s'a=range(1000)' '[x * 2 for x in a]' 10000 loops, best of 3 : 122 usec per loop [root@VM - 0 - 14 - centos ~] # python3 -mtimeit -s'a=range(1000)' 'l = []' 'for i in a: l.append(i * 2)' 1000 loops, best of 3 : 252 usec per loop |
可以看到,使用map+lambda计算的时候,性能是更好的。map函数是由c语言写的,运行的时候不需要通过python解释器,并且内部做了很多优化,因此性能会更好。
以上就是python 匿名函数相关总结的详细内容,更多关于python 匿名函数的资料请关注服务器之家其它相关文章!
原文链接:https://cloud.tencent.com/developer/article/1800965