標籤:外部 結束 bad 順序 pre 過程 int 函數名 TE
變數引用順序
Python 引用變數的順序:當前範圍局部變數->外層範圍變數->當前模組中的全域變數->Python內建變數。
1、 global
global 關鍵字用來在函數或其他局部範圍中使用全域變數,如果不對全域變數進行修改,那麼可以不使用global關鍵字;如果想要在函數或局部範圍中對全域變數進行修改操作,那麼必須在函數或局部範圍中使用global關鍵字進行聲明,否則報錯:UnboundLocalError
count = 1def foo(): global count#如果不加這個那麼就會報錯 count += 1 print(count)foo()
2、 nonlocal
使用方式:閉包,嵌套函數中
如果內建函式想要對外部函數的局部變數進行修改時,需要在內建函式中聲明:
nonlocal <變數名>
nonlocal語句會搜尋當前調用棧中的下一層函數的定義。
def a(): count = 1 def b(): nonlocal count count += 1 print(count) return ba()()
3、 裝飾器
介紹:裝飾器的作用是將被修飾的函數當作參數傳遞給與裝飾器對應的函數,並返回封裝後的被裝飾的函數。裝飾器其實是閉包的一種特殊情況
def a(func): return func@adef b(): passb()
執行:
解析過程:
- 發現@a,那麼就將會執行a(b),返回b
- b(),這一步調用的其實是a返回的同一名的b函數,並且開始執行b函數中的操作
4、 閉包
閉包概念:在電腦科學中,閉包(Closure)是詞法閉包的簡稱,是引用了自由變數的函數,這個被引用的自由變數將和這個函數一同存在,即使已經離開了創造它的環境也不例外。
簡單介紹:可以理解為一個封閉的包裹,這個包裹就是一個函數和函數內部對應的邏輯,包裹裡面的東西就是自由變數,自由變數可以隨著包裹到處遊盪,比如:裝飾器是一個閉包,隨著裝飾器進行傳遞的函數名就是自由變數(函數),裝飾器和內部邏輯結構就是閉包中的函數和內部邏輯。
def func(name): def inner_func(age): print(‘name:’+name+’ age:’+age) return inner_funca = func(‘liyang’)a(20)#>>> name:liyang age:20
上面的例子中:調用func函數時就產生了閉包inner_func,該閉包所持有的自由變數是name。
當函數func的生命週期結束之後,name這個變數依然存在,因為它被閉包引用了,所以不會被回收。
5、 延遲綁定
Python函數的範圍是由代碼決定的,也就是靜態,但是使用是動態,是在執行時確定的
閉包中遇到的問題
fs = [lambda x: x*I for i in range(4)]print(fs[0](1))#>>> 3
期望輸出結果是:0,但是結果卻是:3
這是因為只有在函數foo被執行的時候才會搜尋變數i的值,由於迴圈已結束,i指向最終值3,所以最終會發現結果都是一樣。
使用下面的閉包方式,就會得到期望的結果:
def foo(i): return lambad x: x*ifs = [foo(i) for i in range(4)]print(fs[0](1))
參考:
https://www.cnblogs.com/linxiyue/p/7911916.html
64905070
Python 函數和變數範圍