1. 程式人生 > >python中閉包詳解

python中閉包詳解

ner copy bsp div 執行 gpo 註意 outer 在一起

閉包這個概念好難理解,身邊朋友們好多都稀裏糊塗的,稀裏糊塗的林老冷希望寫下這篇文章能夠對稀裏糊塗的夥伴們有一些幫助~

請大家跟我理解一下,如果在一個函數的內部定義了另一個函數,外部的我們叫他外函數,內部的我們叫他內函數。

閉包:

  在一個外函數中定義了一個內函數,內函數裏運用了外函數的臨時變量,並且外函數的返回值是內函數的引用。這樣就構成了一個閉包。

一般情況下,在我們認知當中,如果一個函數結束,函數的內部所有東西都會釋放掉,還給內存,局部變量都會消失。但是閉包是一種特殊情況,如果外函數在結束的時候發現有自己的臨時變量將來會在內部函數中用到,就把這個臨時變量綁定給了內部函數,然後自己再結束。

  很晦澀很難理解啊!!我們來看一段代碼^.^

技術分享圖片
 1 #閉包函數的實例
 2 # outer是外部函數 a和b都是外函數的臨時變量
 3 def outer( a ):
 4     b = 10
 5     # inner是內函數
 6     def inner():
 7         #在內函數中 用到了外函數的臨時變量
 8         print(a+b)
 9     # 外函數的返回值是內函數的引用
10     return inner
11 
12 if __name__ == ‘__main__‘:
13     # 在這裏我們調用外函數傳入參數5
14     #此時外函數兩個臨時變量 a是5 b是10 ,並創建了內函數,然後把內函數的引用返回存給了demo
15     # 外函數結束的時候發現內部函數將會用到自己的臨時變量,這兩個臨時變量就不會釋放,會綁定給這個內部函數
16     demo = outer(5)
17     # 我們調用內部函數,看一看內部函數是不是能使用外部函數的臨時變量
18     # demo存了外函數的返回值,也就是inner函數的引用,這裏相當於執行inner函數
19     demo() # 15
20 
21     demo2 = outer(7)
22     demo2()#17
技術分享圖片

從上面例子是我寫的一個最簡單的很典型的閉包。我估計如果是初學的小夥伴,可能很多名詞都不明白是什麽意思,沒關系,我把這些名詞按照自己的理解去解釋一下~

1 外函數返回了內函數的引用:

  引用是什麽?在python中一切都是對象,包括整型數據1,函數,其實是對象。

  當我們進行a=1的時候,實際上在內存當中有一個地方存了值1,然後用a這個變量名存了1所在內存位置的引用。引用就好像c語言裏的指針,大家可以把引用理解成地址。a只不過是一個變量名字,a裏面存的是1這個數值所在的地址,就是a裏面存了數值1的引用。

  相同的道理,當我們在python中定義一個函數def demo(): 的時候,內存當中會開辟一些空間,存下這個函數的代碼、內部的局部變量等等。這個demo只不過是一個變量名字,它裏面存了這個函數所在位置的引用而已。我們還可以進行x = demo, y = demo, 這樣的操作就相當於,把demo裏存的東西賦值給x和y,這樣x 和y 都指向了demo函數所在的引用,在這之後我們可以用x() 或者 y() 來調用我們自己創建的demo() ,調用的實際上根本就是一個函數,x、y和demo三個變量名存了同一個函數的引用。

  不知道大家有沒有理解,很晦澀,希望我說明白了我想表達的。

  有了上面的解釋,我們可以繼續說,返回內函數的引用是怎麽回事了。對於閉包,在外函數outer中 最後return inner,我們在調用外函數 demo = outer() 的時候,outer返回了inner,inner是一個函數的引用,這個引用被存入了demo中。所以接下來我們再進行demo() 的時候,相當於運行了inner函數。

  同時我們發現,一個函數,如果函數名後緊跟一對括號,相當於現在我就要調用這個函數,如果不跟括號,相當於只是一個函數的名字,裏面存了函數所在位置的引用。

2 外函數把臨時變量綁定給內函數:

  按照我們正常的認知,一個函數結束的時候,會把自己的臨時變量都釋放還給內存,之後變量都不存在了。一般情況下,確實是這樣的。但是閉包是一個特別的情況。外部函數發現,自己的臨時變量會在將來的內部函數中用到,自己在結束的時候,返回內函數的同時,會把外函數的臨時變量送給內函數綁定在一起。所以外函數已經結束了,調用內函數的時候仍然能夠使用外函數的臨時變量。

  在我編寫的實例中,我兩次調用外部函數outer,分別傳入的值是5和7。內部函數只定義了一次,我們發現調用的時候,內部函數是能識別外函數的臨時變量是不一樣的。python中一切都是對象,雖然函數我們只定義了一次,但是外函數在運行的時候,實際上是按照裏面代碼執行的,外函數裏創建了一個函數,我們每次調用外函數,它都創建一個內函數,雖然代碼一樣,但是卻創建了不同的對象,並且把每次傳入的臨時變量數值綁定給內函數,再把內函數引用返回。雖然內函數代碼是一樣的,但其實,我們每次調用外函數,都返回不同的實例對象的引用,他們的功能是一樣的,但是它們實際上不是同一個函數對象。

閉包中內函數修改外函數局部變量:

  在閉包內函數中,我們可以隨意使用外函數綁定來的臨時變量,但是如果我們想修改外函數臨時變量數值的時候發現出問題了!咋回事捏??!!(哇哇大哭)

  在基本的python語法當中,一個函數可以隨意讀取全局數據,但是要修改全局數據的時候有兩種方法:1 global 聲明全局變量 2 全局變量是可變類型數據的時候可以修改

  在閉包內函數也是類似的情況。在內函數中想修改閉包變量(外函數綁定給內函數的局部變量)的時候:

    1 在python3中,可以用nonlocal 關鍵字聲明 一個變量, 表示這個變量不是局部變量空間的變量,需要向上一層變量空間找這個變量。

    2 在python2中,沒有nonlocal這個關鍵字,我們可以把閉包變量改成可變類型數據進行修改,比如列表。

上代碼!!!

技術分享圖片
 1 #修改閉包變量的實例
 2 # outer是外部函數 a和b都是外函數的臨時變量
 3 def outer( a ):
 4     b = 10  # a和b都是閉包變量
 5     c = [a] #這裏對應修改閉包變量的方法2
 6     # inner是內函數
 7     def inner():
 8         #內函數中想修改閉包變量
 9         # 方法1 nonlocal關鍵字聲明
10         nonlocal  b
11         b+=1
12         # 方法二,把閉包變量修改成可變數據類型 比如列表
13         c[0] += 1
14         print(c[0])
15         print(b)
16     # 外函數的返回值是內函數的引用
17     return inner
18 
19 if __name__ == ‘__main__‘:
20 
21     demo = outer(5)
22     demo() # 6  11
技術分享圖片

從上面代碼中我們能看出來,在內函數中,分別對閉包變量進行了修改,打印出來的結果也確實是修改之後的結果。以上兩種方法就是內函數修改閉包變量的方法。

還有一點需要註意:使用閉包的過程中,一旦外函數被調用一次返回了內函數的引用,雖然每次調用內函數,是開啟一個函數執行過後消亡,但是閉包變量實際上只有一份,每次開啟內函數都在使用同一份閉包變量

上代碼!

技術分享圖片
 1 #coding:utf8
 2 def outer(x):
 3     def inner(y):
 4         nonlocal x
 5         x+=y
 6         return x
 7     return inner
 8 
 9 
10 a = outer(10)
11 print(a(1)) //11
12 print(a(3)) //14
技術分享圖片

兩次分別打印出11和14,由此可見,每次調用inner的時候,使用的閉包變量x實際上是同一個。

閉包有啥用??!!

  很多夥伴很糊塗,閉包有啥用啊??還這麽難懂!

   3.1裝飾器!!!裝飾器是做什麽的??其中一個應用就是,我們工作中寫了一個登錄功能,我們想統計這個功能執行花了多長時間,我們可以用裝飾器裝飾這個登錄模塊,裝飾器幫我們完成登錄函數執行之前和之後取時間。

   3.2面向對象!!!經歷了上面的分析,我們發現外函數的臨時變量送給了內函數。大家回想一下類對象的情況,對象有好多類似的屬性和方法,所以我們創建類,用類創建出來的對象都具有相同的屬性方法。閉包也是實現面向對象的方法之一。在python當中雖然我們不這樣用,在其他編程語言入比如avaScript中,經常用閉包來實現面向對象編程

   3.3實現單利模式!! 其實這也是裝飾器的應用。單利模式畢竟比較高大,,需要有一定項目經驗才能理解單利模式到底是幹啥用的,我們就不探討了。

python中閉包詳解