Python中lambda的学习

在python语法中lambda是一个很单纯的用来简化编程的关键字,使用起来很简单,无非是——lambda x: x+1之类的,但是当它和for、append、list、generator等结合时,却不那么容易就可以读懂代码。

以lambda x: x+1为例,首先,它是一个函数:

def f(x):
    return x+1
print(f(5)) # 6
f = lambda x: x+1
print(f(5)) # 6

这个函数引用时需要传入一个参数,并且有一个返回值。这个参数一般是for x in L之类的传进来,或者直接调用f(3)。

1、先看第一个例子

f = lambda x: x**2
print(f(5)) # 25

结果是25,这里要说明的是lambda x: x**2是一个函数,你如果print(f)的得到的是一个函数的地址,记住它是一个函数。

2、和append搭配、在for循环内部

li = []
for x in range(5):
    li.append(lambda x: x**2)
print(li[0](2)) # 4
print(li[1](3)) # 9

注:li是一个list,但是list里面存的可不是数,而是函数地址,而且是5个x**2的函数,所以无论你是li0还是li1,结果都是4。一般情况下不会这样写程序,因为没什么用途。

这里说一下,看过一个程序这样写,猜测原作者是想让li在运算时append的是数据,或者是以为这样可以让li在调用时n的值不随x变,不管这样,这个程序实际效果和上面一样,x本身在变,n = x写不写没有区别,li内部仍然是5个一样的函数的地址。

li = []
for x in range(5):
    li.append(lambda n=x: n**2)
print(li[0](2)) # 4
print(li[1](3)) # 9

总结一下:lambda在for循环内部,和append搭配时,for循环不是为了给函数传递参数,只是为了生成多个函数。

3、只和append搭配

li = []
li.append(lambda x: x**2)
print(li[0](1)) # 1
print(li[0](3)) # 9
print(li[1](3)) # IndexError: list index out of range

这儿说的是另外一种情况,程序中并没有给出匿名函数lambda的参数,在调用时才会给。而且li仅仅append了一次,所以li内部也仅有一个函数地址。调用时就不会有li[1]这种情况。

补充一种它的变形,说明一下对于这种情况,参数赋初值并无意义。

li = []
li.append(lambda x=5: x**2)
print(li[0](1)) # 1
print(li[0](3)) # 9
print(li[1](3)) # IndexError: list index out of range

4、和append搭配、参数由for循环给出

举个例子

li = []
li.append(lambda :x for x in range(10))
print(next(li[0])()) # 0
print(next(li[0])()) # 1
print(next(li[1])()) # IndexError: list index out of range

此处有大坑,首先你得认出来(lambda :x for x in range(10))这种形式可没有那么简单,这是产生一个生成器最简单的方法,它的返回值是一个generator,所以li内部就存了一个generator。还有此时的函数是没有参数的,等效为:

def f():
    return x

有人会说这个函数有什么意义吗,是没什么意义,但是如果return x**2,其实还是有些意义的。

5、放在[]中、参数由for循环给出

li = []
li = [lambda :x for x in range(10)]
print(li[0]()) # 9
print(li[1]()) # 9

这个函数其实不好理解,首先别看成生成器了,跟它没关系。 lambda :x仍然是一个函数(return x),在没有print(li0)之前它是不会被执行的,一旦运行print(li0),就会输出x的值,那么x是多少呢,显然x在上一句程序里面已经变成9了,所以结果都是9,这里其实是闭包的问题,想避免这个问题,程序就不能写这么简洁了。

li = []
for x in range(5):
    def f():
         return x**2
    li.append(f())# instant run
print(li[0], li[1], li[2], li[3], li[4])

结果是0, 1, 4, 9, 16,是我们想要的,有人会说这儿为什么不把def f()简化一下呢?还真不能简化,比较结果便知:

li = []
for x in range(5):
    li.append(lambda :x**2) # uninstant run
print(li[0](), li[1](), li[2](), li[3](), li[4]())
#16 16 16 16 16

看到区别了吧,f 是一个函数地址,而 f() 是一个函数被执行后的返回值,所以第一个程序可以得到每次循环的 x 值。

6、lambda最常用:和map、reduce、filter等结合用 其实lambda最常用的还是和map、reduce、filter这些高级函数结合使用,不过那个时候就把它当做一个函数,而且格式相对固定,具体使用就看高级函数的使用规则,较为简单,就不展开。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Python数据科学

Python 内建函数大全

Python 解释器内置了许多函数和类型,列表如下(按字母排序)(省略了几个我没用过或者不常用的)。

2533
来自专栏Python

Python常见数据结构整理 Python常见数据结构整理

Python常见数据结构整理 Python中常见的数据结构可以统称为容器(container)。序列(如列表和元组)、映射(如字典)以及集合(set)是三类主要...

1947
来自专栏Python自动化测试

python的字典学习(六)

本博客主要说明python的字典基本的使用,在python中,字典使用的关键字是dict,使用的是{},下面我们通过一个具体的代码来看python字典类对...

1193
来自专栏逻辑熊猫带你玩Python

Python | 6大数据类型方法归纳总结(下)

可以直接使用list()创建一个新的列表,或者,使用list()将一个对象转换成列表。

2073
来自专栏逻辑熊猫带你玩Python

Python | 6大数据类型方法归纳总结(中)

可以直接使用tuple()创建一个新的元组,或者,使用tuple()将一个对象转换成元组。

1674
来自专栏编程

JavaSE基础:包装类

包装类 1.为什么需要包装类? Java在设计之初有一个基本原则:一切皆对象,一切的操作都要求用对象的形式进行描述。但是这里面就会出现一个矛盾,基本数据类型不是...

1996
来自专栏null的专栏

挑战数据结构与算法面试题——统计上排数在下排出现的次数

题目来源“数据结构与算法面试题80道”。在此给出我的解法,如你有更好的解法,欢迎留言。 ? 分析: 本题应该是一个确定的问题,即上排的是个数是题目中给定的...

3216

Python3 字符串操作

字符串是Python中最基本的数据类型之一,用于表示文本数据。几乎每个应用程序都涉及使用字符串,Python的str类提供了许多方法来简化字符串操作。

1244
来自专栏星回的实验室

js重修课[四]:函数

函数有两种定义方法:定义表达式如var f = function(){};和声明语句如function f(){}。须知在变量提前这一现象中,声明语句可被提前,...

1062
来自专栏用户2442861的专栏

剑指offer 36 数组中的逆序对

转载请注明出处:http://blog.csdn.net/ns_code/article/details/27520535

1261

扫码关注云+社区

领取腾讯云代金券