同為腳本語言,python和Javascript具有相似的變量作用域,不像php,函數(shù)的內(nèi)部的所有變量和外部都是隔絕的,也就是說,函數(shù)要想處理其外部的數(shù)據(jù),必須使用參數(shù)把需要處理的數(shù)據(jù)傳遞進來(使用global關(guān)鍵詞這里不討論),而python和Javascript不同,如果在函數(shù)聲明變量,它會逐級網(wǎng)上查找,直到返回著個值或者未定義。
那么這樣說,python的閉包應(yīng)該很簡單了,像javascript一樣,我們編寫類似的代碼:
def func1(): a = 1 def func2(): a = a + 1 return a return func2 re=func1() print re() print re()
但是,實際情況是,結(jié)果并沒有出現(xiàn)我們預(yù)期中的打印出2和3,反而出現(xiàn)了這樣的錯誤:”UnboundLocalError: local variable ‘a(chǎn)' referenced before assignment”(局部變量a賦值之前被引用)。為什么會出現(xiàn)這樣的問題,我們先看看js是如果實現(xiàn)這個閉包的:
<script> function func1(){ var a=1; function func2(){ a=a+1; return a; } return func2; } re=func1(); console.log(re()); console.log(re()); </script>
上面這段代碼的運行結(jié)果如我們所料,輸入2和3。注意一下這段程序的第5行,如果我在前面加上一個var,這段程序運行的結(jié)果是什么樣的呢?最終結(jié)果是輸入了兩個“NaN”,在火狐的開發(fā)者平臺上,找到了關(guān)于var這樣的描述:
Declares a variable, optionally initializing it to a value.
The scope of a variable declared with var is the enclosing function or, for variables declared outside a function, the global scope (which is bound to the global object).
意思是說,var是用來聲明局部變量的,上面的例子中,如果用var a=a+1,這時候的a就已經(jīng)是func2中的局部變量,而不會從func1中繼承,所以最后會出現(xiàn)NaN的結(jié)果。
讓我們回到python的這個閉包上來,這個錯誤提示的意思也是說a是個局部變量,實際上,python規(guī)定所有在賦值語句左面的變量都是局部變量,這個a在等號左邊,所以成了一個局部的變量,導致我訪問不到func1中的a。這個問題怎么解決呢?如果是在python3.0以上,在a=a+1之前,可以用nonloacal a來指定a不為局部變量。3.0以下的版本不支持nonloacal關(guān)鍵字,我們可以這樣做:
def func1(): a = [1] def func2(): a[0] = a[0] + 1 return a[0] return func2 re=func1() print re() print re()
運行結(jié)果如我們所料,打印出了2和3。從python和Javascript閉包的例子,要了解python和js變量聲明,變量作用域的相似和不同之處。
更多信息請查看IT技術(shù)專欄