Python定义匿名函数,以及如何捕获变量值?

  1. 定义匿名函数

问题

你想为sort()操作创建一个很短的回调函数,但又不想用 def 去写一个单行函数,而是希望通过某个快捷方式以内联方式来创建这个函数。

解决方案

当一些函数很简单,仅仅只是计算一个表达式的值的时候,就可以使用 lambda 表 达式来代替了。比如:

>>> add = lambda x, y: x + y
>>> add(2,3)
5
>>> add('hello', 'world')
'helloworld'
>>>

这里使用的 lambda 表达式跟下面的效果是一样的:

>>> def add(x, y):
...      return x + y
...
>>> add(2,3)
5
>>>

lambda 表达式典型的使用场景是排序或数据 reduce 等:

>>> names = ['David Beazley', 'Brian Jones',
...      'Raymond Hettinger', 'Ned Batchelder']
>>> sorted(names, key=lambda name: name.split()[-1].lower())
['Ned Batchelder', 'David Beazley', 'Raymond Hettinger', 'Brian Jones']
>>>

讨论

尽管 lambda 表达式允许你定义简单函数,但是它的使用是有限制的。你只能指定单个表达式,它的值就是最后的返回值。也就是说不能包含其他的语言特性了,包括多个语句、条件表达式、迭代以及异常处理等等。 你可以不使用 lambda 表达式就能编写大部分 python 代码。但是,当有人编写大量计算表达式值的短小函数或者需要用户提供回调函数的程序的时候,你就会看到 lambda表达式的身影了。


  1. 匿名函数捕获变量值

问题

你用 lambda 定义了一个匿名函数,并想在定义时捕获到某些变量的值。

解决方案

先看下下面代码的效果:

>>> x = 10
>>> a = lambda y: x + y
>>> x = 20
>>> b = lambda y: x + y
>>>

现在问你,a(10)和b(10)返回的结果是什么?如果你认为结果20 和30,那么你就错了:

>>> a(10)
30
>>> b(10)
30
>>>

这其中的奥妙在于 lambda 表达式中的 x 是一个自由变量,在运行时绑定值,而不是定义时就绑定,这跟函数的默认值参数定义是不同的。因此,在调用这个lambda 表达式的时候,x 的值是执行时的值。例如:

>>> x = 15
>>> a(10)
25
>>> x = 3
>>> a(10)
13
>>>

如果你想让某个匿名函数在定义时就捕获到值,可以将那个参数值定义成默认参 数即可,就像下面这样:

>>> x = 10
>>> a = lambda y, x=x: x + y
>>> x = 20
>>> b = lambda y, x=x: x + y
>>> a(10)
20
>>> b(10)
30
>>>

讨论

在这里列出来的问题是新手很容易犯的错误,有些新手可能会不恰当的使用 lambda 表达式。比如,通过在一个循环或列表推导中创建一个 lambda 表达式列表,并期望函数能在定义时就记住每次的迭代值。例如:

>>> funcs = [lambda x: x+n for n in range(5)]
>>> for f in funcs:
...      print(f(0))
...
4
4
4
4
4
>>>

但是实际效果是运行时n的值为迭代的最后一个值。现在我们用另一种方式修改 一下:

>>> funcs = [lambda x, n=n: x+n for n in range(5)]
>>> for f in funcs:
...      print(f(0))
...
0
1
2
3
4
>>>

通过使用函数默认值参数形式,lambda 函数在定义时就能绑定到值。

展开阅读全文

页面更新:2024-03-25

标签:函数   定义   表达式   绑定   变量   解决方案   参数   效果   新手   代码

1 2 3 4 5

上滑加载更多 ↓
推荐阅读:
友情链接:
更多:

本站资料均由网友自行发布提供,仅用于学习交流。如有版权问题,请与我联系,QQ:4156828  

© CopyRight 2020-2024 All Rights Reserved. Powered By 71396.com 闽ICP备11008920号-4
闽公网安备35020302034903号

Top