Python閉包

轉自慕課網

python中閉包

在函數內部定義的函數和外部定義的函數是一樣的,只是他們無法被外部訪問:

def g():
    print 'g()...'

def f():
    print 'f()...'
    return g

 g 的定義移入函數 f 內部,防止其他代碼調用 g

def f():
    print 'f()...'
    def g():
        print 'g()...'
    return g

但是,考察上一小節定義的 calc_sum 函數:

def calc_sum(lst):
    def lazy_sum():
        return sum(lst)
    return lazy_sum

注意: 發現沒法把 lazy_sum 移到 calc_sum 的外部,因爲它引用了 calc_sum 的參數 lst

像這種內層函數引用了外層函數的變量(參數也算變量),然後返回內層函數的情況,稱爲閉包(Closure)

閉包的特點是返回的函數還引用了外層函數的局部變量,所以,要正確使用閉包,就要確保引用的局部變量在函數返回後不能變。舉例如下:

# 希望一次返回3個函數,分別計算1x1,2x2,3x3:
def count():
    fs = []
    for i in range(1, 4):
        def f():
             return i*i
        fs.append(f)
    return fs

f1, f2, f3 = count()

你可能認爲調用f1(),f2()和f3()結果應該是1,4,9,但實際結果全部都是 9(請自己動手驗證)。

原因就是當count()函數返回了3個函數時,這3個函數所引用的變量 i 的值已經變成了3。由於f1、f2、f3並沒有被調用,所以,此時他們並未計算 i*i,當 f1 被調用時:

>>> f1()
9     # 因爲f1現在才計算i*i,但現在i的值已經變爲3

因此,返回函數不要引用任何循環變量,或者後續會發生變化的變量。

下面給出正確的代碼:

def count():
    fs = []
    for i in range(1, 4):
        def f(j):
            def g():
                return j*j
            return g
        r = f(i)
        fs.append(r)
    return fs
f1, f2, f3 = count()
print f1(), f2(), f3()

轉成Javascript代碼如下:
function count(){
	var funArr = new Array();
	var arr = [1, 2, 3];
	for(var i in arr){
		function f(j){
			function g(){
				return j * j;
			}
			return g;
		}
		r = f(arr[i]);
		funArr.push(r);
	}
	return funArr;
}
var rfArr = count();
for(var rf in rfArr){
	console.log(rfArr[rf]());
}



發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章