7.7 匿名函數捕獲變量值

2018-02-24 15:26 更新

問題

你用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的值是執(zhí)行時的值。例如:

>>> 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表達式。比如,通過在一個循環(huán)或列表推導中創(chuàng)建一個lambda表達式列表,并期望函數能在定義時就記住每次的迭代值。例如:

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

但是實際效果是運行是n的值為迭代的最后一個值?,F在我們用另一種方式修改一下:

>>> 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函數在定義時就能綁定到值。

以上內容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號