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