python 函式式程式設計 閉包,返回乙個函式

2022-08-18 22:09:25 字數 2574 閱讀 7135

#使用生成器

def createcounter():

def count():

d=0while true:

d+=1

yield d

c=count()

def f2():

return next(c)

return f2

#使用nonlocal關鍵字

def createcounter():

i=0def count():

nonlocal i

i+=1

return i

return count

通常我們定義的函式都是返回函式的執行結果,但是假如我們不需要讓這個函式立即執行,而是在後面某些情況下才執行,我們就可以定義乙個函式,讓這個函式返回我們需要執行的函式。

假設我們需要返回的函式是這樣的:

def calc_sum(*args):

ax = 0

for n in args:

ax = ax + n

return ax

那麼返回上面這個函式的函式就可以這樣定義:

def lazy_sum(*args):

def sum():

ax = 0

for n in args:

ax = ax + n

return ax

return sum

定義乙個返回函式的函式,只需要把原函式的引數挪到外面的函式中,然後原封不動的將裡面的函式增加乙個縮排使其符合縮排規則,最後把函式名在外面的函式中返回就好了

當我們呼叫lazy_sum()時,返回的並不是求和結果,而是求和函式:

>>> f = lazy_sum(1, 3, 5, 7, 9)

>>> f

.sum at 0x101c6ed90>

當我們真正想要執行函式的時候,就加上括號,讓函式執行

>>> f()

25

注意閉包不能接受引數,因為返回的是函式名,無法傳遞引數

在這個例子中,我們在函式lazy_sum中又定義了函式sum,並且,內部函式sum可以引用外部函式lazy_sum的引數和區域性變數,當lazy_sum返回函式sum時,相關引數和變數都儲存在返回的函式中,這種稱為「閉包(closure)」的程式結構擁有極大的威力。

需要注意的是,當我們呼叫lazy_sum()時,每次呼叫都會返回乙個新的函式,每個函式的呼叫間不受影響。即使傳入相同的引數:

>>> f1 = lazy_sum(1, 3, 5, 7, 9)

>>> f2 = lazy_sum(1, 3, 5, 7, 9)

>>> f1==f2

false

閉包的另乙個問題,返回函式不要引用任何迴圈變數,或者會發生變化的變數

下面的返回的函式執行結果是一樣的,並不是我們認為的1,4,9,而都是9

def count():

fs =

for i in range(1, 4):

def f():

return i*i

return fs

f1, f2, f3 = count()

>>> f1()

9>>> f2()

9>>> f3()

9

這是因為,返回的函式並沒有立刻執行,而是直到呼叫了f()才執行。等到3個函式都返回時,它們所引用的變數i已經變成了3,因此最終結果為9

返回的函式在其定義內部引用了區域性變數args,所以,當乙個函式返回了乙個函式後,其內部的區域性變數還被新函式引用,所以,閉包用起來簡單,實現起來可不容易。

返回閉包時牢記一點:返回函式不要引用任何迴圈變數,或者後續會發生變化的變數。

如果一定要引用迴圈變數呢?

方法是再建立乙個函式,用該函式的引數繫結迴圈變數當前的值,無論該迴圈變數後續如何更改,已繫結到函式引數的值不變:

def count():

def f(j):#新建乙個函式

def g():

return j*j

return g

fs =

for i in range(1, 4):

return fs

>>> f1, f2, f3 = count()

>>> f1()

1>>> f2()

4>>> f3()

9

缺點是**較長,可利用lambda函式縮短**。

python 函式式程式設計 閉包,返回乙個函式

作業 使用生成器 defcreatecounter def count d 0while true d 1yield d c count deff2 return next c return f2 使用nonlocal關鍵字 defcreatecounter i 0def count nonloca...

Python函式式程式設計之閉包

def div fun n def div check fun x return x n 0 return div check fun這是什麼?外層函式中巢狀了乙個函式,然後外層函式將內層函式作為返回值進行返回,同時,返回函式中的計算也擁有了外層函式的變數n,這種將外部函式的引數和自身的區域性變數儲...

函式式程式設計 閉包

def curve pie a 25 defcurve x return a pow x,2 return curve f curve pie print f 2 輸出結果 ans 100 檢驗函式是否閉包 print f.closure 環境變數 a 25 print f.closure 0 ce...