Python入门系列28 - 进阶必修之匿名函数与高阶函数

小黄用python

共 2474字,需浏览 5分钟

 ·

2019-12-19 23:22

Python入门系列28

8d8bfedaff336a33392f534d61517f5c.webp

进阶必修之匿名函数与高阶函数


本篇阅读时间约为 7 分钟。


1

前言


嗨!大家好哇,让我们继续回归Python入门系列之路,本章为进阶篇,介绍的是匿名函数与高阶函数。


2

匿名函数之 lambda 表达式


见名知意,所谓的匿名函数就是没有名字的函数。我们平时所写的函数都是带有名字和参数的,并且有返回值,而匿名函数的写法却非常简洁,对于初学者而言,代码的阅读性可能会较差一些,但是多加实践,看习惯了之后就能明白所谓的lambda表达式是什么意思了!接下来还是用示例说话,场景是这样的:有个函数,传入两个参数a和b,在函数内部将他们做差,并将结果返回。


示例1 - 普通函数:


def sub(a, b):
    return a - b

result = sub(12)
print(result)

>>> -1


示例2 - 匿名函数:


sub = lambda a, b: a - b
result = sub(12)
print(result)


142471a3e35fd45905446743dee12333.webp

解释一下上面的案例,使用 python 中的 lambda 关键词来定义匿名函数,参数在冒号前面,表达式在冒号后面,将整个 lambda 表达式赋予一个变量sub,通过调用sub传入参数得到结果result,与普通函数得到一致的结果。


匿名函数的写法,通过关键词 lambda 来表示匿名,基于数学中的λ演算得名。lambda表达式完整的写法公式如下:


lambda parameter_list : expression


什么意思呢?通过 lambda 后面跟上传入的参数,加上冒号,以及冒号后面的表达式组成了匿名函数,注意:冒号后面一定要是表达式!


表达式意味着,如果我们写成赋值语句或者for循环语句等非表达式语句,那么 lambda 表达式则会报错!如下:

9166155d9c8f5a017cd42650569dba82.webp


所以由此引出了匿名函数 lambda 中常用的三元表达式


示例3 - Python的三元表达式:

假设现在有两个变量,a = 1,b = 2,请使用一行代码来判断 a 小于 b ,若 a 小于 b 为真,则将 a 的值返回。


a = 1
b = 2
result = a if a < b else b
print(result)


09129ffcfa5442c61cebdc51fa56853d.webp

result 这一行就是所谓的 Python 中的三元表达式,用中文描述一下就是:

条件为真时返回的结果 if 条件判断 else 条件为假时返回的结果


3

高阶函数


高阶函数,Python中自带了一些高阶特性的函数,在某些特定场景下非常实用,这里举几个常用的,当然也会结合具体场景去说,毕竟要结合使用不能滥用!


1. map 函数

map 函数,map中文意思是地图的意思,但是在程序中代表映射的意思。高中学过的映射,不知大家是否还记得!跟那个意思差不多。。。


场景如下:现在有一个 list_a ,它其中的包含着 1~6 的数字,请将此集合中的数字都乘以 2 ,最终得到的结果组成一个新的 list_b。


list_a = [123456]
xxxxx
得到 list_b


不使用 map ,你会怎么做呢?思考一下,然后再看下面的答案:

list_a = [123456]
list_b = []

def double_fun(a):
    return 2 * a

for a in list_a:
    list_b.append(double_fun(a))

print(list_b)


fff00ca2cb94b62e396ac26040786a86.webp

通过 for 循环取出 list 每个值,在对其调用乘2的函数,最后将结果进行拼接,得到 list_b 。


使用 map 函数之前,先了解下 map 函数语法:

map(function, iterable,...)

map() 会根据提供的函数对指定序列做映射。

第一个参数 function 以参数序列中的每一个元素调用 function 函数,返回包含每次 function 函数返回值的新列表。


list_a = [123456]

def double_fun(a):
    return 2 * a

result = map(double_fun, list_a)
print(result)
print(type(result))
print(list(result))


d960b1a1021e4e2544c30a02b87d928f.webp


可以看到,素质三连print!得到的结果,map返回的结果是一个 map 对象,而它本身的类型就是 map,将其转化为 list,可以得到与不用 map 函数时的一样结果!


实际上,如果会活学活用的话,还可以将上述代码进行缩减,用我们刚学过的 lambda 表达式来完成!


list_a = [123456]


# def double_fun(a):
#     return 2 * a

result = map(lambda a: 2 * a, list_a)
print(list(result))


将普通函数注释掉,使用匿名函数来完成,实际上就3行代码即可完成,如下:

b1ee604d819d8b7f33f7cfc93545fe15.webp


如果有多个参数作为匿名函数的入参如何写呢?

可以使用 pycharm 自带的源码查看功能,查看 map 函数的定义,按住 Ctrl 点击 map 即可:

7fed258949df2bb97c5db956a9984ca4.webp


map 第一个参数传入函数名称,第二个传入的是一个可以迭代的对象,例如list,而用红框标出来的其中包括了一个 * 号,在Python中参数前面跟一个星号代表多个形参的意思。所以我们可以写法如下,当有两个相同 list 时,多次传入 map 函数中即可:


list_a = [123456]
list_b = [123456]
result = map(lambda a, b: 2 * a + 2 * b, list_a, list_b)
print(list(result))


d2caf350443a61d852bad15e8b2cf3b3.webp


现在是相同的个数的 list ,如果要是不同个数内容的 list 会如何?做个实验,将 list_b 的个数改为 10 个观察下:


list_a = [123456]
list_b = [12345678910]
result = map(lambda a, b: 2 * a + 2 * b, list_a, list_b)
print(list(result))


98e5c01680419a4029b8bad37aed58ce.webp


可以看到,即使两个不同内容的list进行map计算,最终结果也会根据内容少的list得到最终结果,也就是只进行前6位的计算。


2. reduce 函数

reduce 函数不能直接进行引用,在 Python3 中,reduce() 函数已经被从全局命名空间里移除了,它现在被放置在 functools 模块里,如果想要使用它,则需要通过引入 functools 模块来调用 reduce() 函数,还是先来看下 reduce 函数的语法使用:

reduce(function, iterable[, initializer])

function -- 函数,有两个参数

iterable -- 可迭代对象

initializer -- 可选,初始参数


函数将一个数据集合(链表,元组等)中的所有数据进行下列操作:用传给 reduce 中的函数 function(有两个参数)先对集合中的第 1、2 个元素进行操作,得到的结果再与第三个数据用 function 函数运算,最后得到一个结果。


老规矩,点进去看源码,其中有官方的文档注释:

52f55a32fcdb3b8c5ebfc88fbcc16fa0.webp

咱们就用它注释中的例子来实践一下,作为 reduce 的第一个函数,入参必须是两个参数,否则会报错,例如下面:

c105a3d9fdefa3f2f62cb3fc4359256a.webp


改成官方例子:

from functools import reduce

result = reduce(lambda x, y: x + y, [12345])
print(result)
print(type(result))


cabf922007586d6737475bfe2022e12c.webp


素质二连,print出结果,可以看到这次与 map 函数不同,返回的结果是一个计算后的整数 15 !这个 reduce 函数是怎么算出来的呢?仔细看过注释的同学一定知道怎么来的了,实际上就是这样来的:


 reduce(lambda x, y: x+y, [12345])  --->  ((((1+2)+3)+4)+5)


第一次计算时,x = 1,y = 2,于是将 1 + 2 = 3 的结果作为下一次调用时入参的x,而第二次计算时的 y 则是列表中的数字3 ......以此类推

(1+2) --> ((1+2)+3) --> ....  --> ((((1+2)+3)+4)+5)


reduce 函数是一个连续计算的函数!


再来看下 reduce 函数的第三个参数,初始化第一个入参值:

from functools import reduce

result = reduce(lambda x, y: x + y, [12345], 10)
print(result)


833ad508604521cae7f11f69ef0766af.webp

这个计算的顺序则是第一次 x = 10,y = 1,将 x+y 的值 11 赋予第二次计算的 x 作为入参,第二次计算时的 y = 2,第二次计算结果等于 11 + 2 = 13 .... 以此类推。

(10+1) --> (10+1)+2) --> ((10+1)+2)+3) .... --> ((((10+1)+2)+3)+4)+5)


3. filter 函数

filter 函数,这个英文单词的中文含义是过滤的意思。依然是看下官方语法:

filter(function, iterable)

function -- 判断函数。

iterable -- 可迭代对象。


filter() 函数用于过滤序列,过滤掉不符合条件的元素,返回由符合条件元素组成的新列表。

该接收两个参数,第一个为函数,第二个为序列,序列的每个元素作为参数传递给函数进行判,然后返回 True 或 False,最后将返回 True 的元素放到新列表中。


示例,请将下面的 list 中为小写字母的字符串提取出来,生成一个新的列表:


list_word = ['a''A''B''c''D''E''f']


思考一下再看答案哟:


list_word = ['a''A''B''c''D''E''f']
result = filter(lambda word: word if word.islower() else False, list_word)
print(result)
print(list(result))

>>> 0x000001D82D743080>
>>> ['a''c''f']


7d18939e928e69d0809bebf38b01437d.webp

结果如上图,通过字符串的自带函数,islower() 可以判断字符串是否是小写字母,若是小写字母则返回一个 True ,可以看到笔者的 lambda 表达式用了一个三元表达式的写法,来过滤掉了原列表的大写字母,是不是很简洁呢!毕竟人生苦短,我用Python啊!


4

map - reduce 扩展知识


map / reduce 实际上是大数据中的一个编程模型,其中 map 代表映射的意思,reduce 代表归约的意思,而这个模型主要是用来进行并行计算的!可以用一张图来看生动的看下 hadoop (Hadoop是一个由Apache基金会所开发的分布式系统基础架构。)中的 map / reduce:


13b72de716a0da73a0cd5228ab1f9fdd.webp

做三明治,用到了各种蔬菜,首先我们先将每个蔬菜切碎,通过map 映射成碎片,在使用 reduce 进行并行计算使之合体!


5

总结


说一些学习上的技巧,对于一个新的函数,首先学习看它的语法,第一时间可以去找官方文档看,比如点进去看它的官方注释,如果英文看不懂的话呢,可以去菜鸟教程上搜下,上面也是有不少官方文档的定义,都是中文教程。写到现在,大部分Python小课堂上的小实例都是非常简单的,入门与进阶通俗易懂,但是不要小瞧这些小 demo ,如果你真的把它们都弄明白了,就像前期种下一颗种子一样,稳定根基之后必将作为长成参天大树的必要前提!


至此完!

9856f6341f575bc1c17c7eb59b794b41.webp


浏览 39
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报