first commit

This commit is contained in:
张乾
2024-10-16 06:37:41 +08:00
parent 633f45ea20
commit 206fad82a2
3590 changed files with 680090 additions and 0 deletions

View File

@ -0,0 +1,228 @@
因收到Google相关通知网站将会择期关闭。相关通知内容
10 简约不简单的匿名函数
你好,我是景霄。
上一节我们一起学习了Python中的“常规”函数用途十分广泛。不过除了常规函数你应该也会在代码中见到一些“非常规”函数它们往往很简短就一行并且有个很酷炫的名字——lambda没错这就是匿名函数。
匿名函数在实际工作中同样举足轻重正确地运用匿名函数能让我们的代码更简洁、易读。这节课我们继续Python的函数之旅一起来学习这个简约而不简单的匿名函数。
匿名函数基础
首先,什么是匿名函数呢?以下是匿名函数的格式:
lambda argument1, argument2,... argumentN : expression
我们可以看到匿名函数的关键字是lambda之后是一系列的参数然后用冒号隔开最后则是由这些参数组成的表达式。我们通过几个例子看一下它的用法
square = lambda x: x**2
square(3)
9
这里的匿名函数只输入一个参数x输出则是输入x的平方。因此当输入是3时输出便是9。如果把这个匿名函数写成常规函数的形式则是下面这样
def square(x):
return x**2
square(3)
9
可以看到匿名函数lambda和常规函数一样返回的都是一个函数对象function object它们的用法也极其相似不过还是有下面几点区别。
第一lambda是一个表达式expression并不是一个语句statement
所谓的表达式就是用一系列“公式”去表达一个东西比如x + 2、 x**2等等
而所谓的语句则一定是完成了某些功能比如赋值语句x = 1完成了赋值print语句print(x)完成了打印,条件语句 if x < 0:完成了选择功能等等
因此lambda可以用在一些常规函数def不能用的地方比如lambda可以用在列表内部而常规函数却不能
[(lambda x: x*x)(x) for x in range(10)]
# 输出
[0, 1, 4, 9, 16, 25, 36, 49, 64, 81]
再比如lambda可以被用作某些函数的参数而常规函数def也不能
l = [(1, 20), (3, 0), (9, 10), (2, -1)]
l.sort(key=lambda x: x[1]) # 按列表中元组的第二个元素排序
print(l)
# 输出
[(2, -1), (3, 0), (9, 10), (1, 20)]
常规函数def必须通过其函数名被调用因此必须首先被定义但是作为一个表达式的lambda返回的函数对象就不需要名字了
第二lambda的主体是只有一行的简单表达式并不能扩展成一个多行的代码块
这其实是出于设计的考虑Python之所以发明lambda就是为了让它和常规函数各司其职lambda专注于简单的任务而常规函数则负责更复杂的多行逻辑关于这点Python之父Guido van Rossum曾发了一篇文章解释你有兴趣的话可以自己阅读
为什么要使用匿名函数
理论上来说Python中有匿名函数的地方都可以被替换成等价的其他表达形式一个Python程序是可以不用任何匿名函数的不过在一些情况下使用匿名函数lambda可以帮助我们大大简化代码的复杂度提高代码的可读性
通常我们用函数的目的无非是这么几点
减少代码的重复性
模块化代码
对于第一点如果你的程序在不同地方包含了相同的代码那么我们就会把这部分相同的代码写成一个函数并为它取一个名字方便在相对应的不同地方调用
对于第二点如果你的一块儿代码是为了实现一个功能但内容非常多写在一起降低了代码的可读性那么通常我们也会把这部分代码单独写成一个函数然后加以调用
不过再试想一下这样的情况你需要一个函数但它非常简短只需要一行就能完成同时它在程序中只被调用一次而已那么请问你还需要像常规函数一样给它一个定义和名字吗
答案当然是否定的这种情况下函数就可以是匿名的你只需要在适当的地方定义并使用就能让匿名函数发挥作用了
举个例子如果你想对一个列表中的所有元素做平方操作而这个操作在你的程序中只需要进行一次用lambda函数可以表示成下面这样
squared = map(lambda x: x**2, [1, 2, 3, 4, 5])
如果用常规函数则表示为这几行代码
def square(x):
return x**2
squared = map(square, [1, 2, 3, 4, 5])
这里我简单解释一下函数map(function, iterable)的第一个参数是函数对象第二个参数是一个可以遍历的集合它表示对iterable的每一个元素都运用function这个函数两者一对比我们很明显地发现lambda函数让代码更加简洁明了
再举一个例子在Python的Tkinter GUI应用中我们想实现这样一个简单的功能创建显示一个按钮每当用户点击时就打印出一段文字如果使用lambda函数可以表示成下面这样
from tkinter import Button, mainloop
button = Button(
text='This is a button',
command=lambda: print('being pressed')) # 点击时调用lambda函数
button.pack()
mainloop()
而如果我们用常规函数def那么需要写更多的代码
from tkinter import Button, mainloop
def print_message():
print('being pressed')
button = Button(
text='This is a button',
command=print_message) # 点击时调用lambda函数
button.pack()
mainloop()
显然运用匿名函数的代码简洁很多也更加符合Python的编程习惯
Python函数式编程
最后我们一起来看一下Python的函数式编程特性这与我们今天所讲的匿名函数lambda有着密切的联系
所谓函数式编程是指代码中每一块都是不可变的immutable都由纯函数pure function的形式组成这里的纯函数是指函数本身相互独立互不影响对于相同的输入总会有相同的输出没有任何副作用
举个很简单的例子比如对于一个列表我想让列表中的元素值都变为原来的两倍我们可以写成下面的形式
def multiply_2(l):
for index in range(0, len(l)):
l[index] *= 2
return l
这段代码就不是一个纯函数的形式因为列表中元素的值被改变了如果我多次调用multiply_2()这个函数那么每次得到的结果都不一样要想让它成为一个纯函数的形式就得写成下面这种形式重新创建一个新的列表并返回
def multiply_2_pure(l):
new_list = []
for item in l:
new_list.append(item * 2)
return new_list
函数式编程的优点主要在于其纯函数和不可变的特性使程序更加健壮易于调试debug和测试缺点主要在于限制多难写当然Python不同于一些语言比如Scala它并不是一门函数式编程语言不过Python也提供了一些函数式编程的特性值得我们了解和学习
Python主要提供了这么几个函数map()、filter()和reduce()通常结合匿名函数lambda一起使用这些都是你需要掌握的东西接下来我逐一介绍
首先是map(function, iterable)函数前面的例子提到过它表示对iterable中的每个元素都运用function这个函数最后返回一个新的可遍历的集合比如刚才列表的例子要对列表中的每个元素乘以2那么用map就可以表示为下面这样
l = [1, 2, 3, 4, 5]
new_list = map(lambda x: x * 2, l) # [2 4 6 8 10]
我们可以以map()函数为例看一下Python提供的函数式编程接口的性能还是同样的列表例子它还可以用for循环和list comprehension目前没有统一中文叫法你也可以直译为列表理解等实现我们来比较一下它们的速度
python3 -mtimeit -s'xs=range(1000000)' 'map(lambda x: x*2, xs)'
2000000 loops, best of 5: 171 nsec per loop
python3 -mtimeit -s'xs=range(1000000)' '[x * 2 for x in xs]'
5 loops, best of 5: 62.9 msec per loop
python3 -mtimeit -s'xs=range(1000000)' 'l = []' 'for i in xs: l.append(i * 2)'
5 loops, best of 5: 92.7 msec per loop
你可以看到map()是最快的因为map()函数直接由C语言写的运行时不需要通过Python解释器间接调用并且内部做了诸多优化所以运行速度最快
接下来来看filter(function, iterable)函数它和map函数类似function同样表示一个函数对象filter()函数表示对iterable中的每个元素都使用function判断并返回True或者False最后将返回True的元素组成一个新的可遍历的集合
举个例子比如我要返回一个列表中的所有偶数可以写成下面这样
l = [1, 2, 3, 4, 5]
new_list = filter(lambda x: x % 2 == 0, l) # [2, 4]
最后我们来看reduce(function, iterable)函数它通常用来对一个集合做一些累积操作
function同样是一个函数对象规定它有两个参数表示对iterable中的每个元素以及上一次调用后的结果运用function进行计算所以最后返回的是一个单独的数值
举个例子我想要计算某个列表元素的乘积就可以用reduce()函数来表示
l = [1, 2, 3, 4, 5]
product = reduce(lambda x, y: x * y, l) # 1*2*3*4*5 = 120
当然类似的filter()和reduce()的功能也可以用for循环或者list comprehension来实现
通常来说在我们想对集合中的元素进行一些操作时如果操作非常简单比如相加累积这种那么我们优先考虑map()、filter()、reduce()这类或者list comprehension的形式至于这两种方式的选择
在数据量非常多的情况下比如机器学习的应用那我们一般更倾向于函数式编程的表示因为效率更高
在数据量不多的情况下并且你想要程序更加Pythonic的话那么list comprehension也不失为一个好选择
不过如果你要对集合中的元素做一些比较复杂的操作那么考虑到代码的可读性我们通常会使用for循环这样更加清晰明了
总结
这节课我们一起学习了Python中的匿名函数lambda它的主要用途是减少代码的复杂度需要注意的是lambda是一个表达式并不是一个语句它只能写成一行的表达形式语法上并不支持多行匿名函数通常的使用场景是程序中需要使用一个函数完成一个简单的功能并且该函数只调用一次
其次我们也入门了Python的函数式编程主要了解了常见的map()fiilter()和reduce()三个函数并比较了它们与其他形式for循环comprehension的性能显然它们的性能效率是最优的
思考题
最后我想给你留下两道思考题
第一问如果让你对一个字典根据值进行由高到底的排序该怎么做呢以下面这段代码为例你可以思考一下
d = {'mike': 10, 'lucy': 2, 'ben': 30}
第二问在实际工作学习中你遇到过哪些使用匿名函数的场景呢
欢迎在留言区写下你的答案想法与我讨论也欢迎你把这篇文章分享给你的同事朋友