在函數(shù)內(nèi)部定義的函數(shù)和外部定義的函數(shù)是一樣的,只是他們無法被外部訪問:
def g(): print('g()...') def f(): print('f()...') return g
將g的定義移入函數(shù) f 內(nèi)部,防止其他代碼調(diào)用 g:
def f(): print('f()...') def g(): print('g()...') return g
但是,考察上一小節(jié)定義的 calc_sum 函數(shù):
def calc_sum(list_): def lazy_sum(): return sum(list_) return lazy_sum
注意: 發(fā)現(xiàn)沒法把 lazy_sum 移到 calc_sum 的外部,因為它引用了 calc_sum 的參數(shù) list_。
像這種內(nèi)層函數(shù)引用了外層函數(shù)的變量(參數(shù)也算變量),然后返回內(nèi)層函數(shù)的情況,稱為閉包(Closure)。
閉包的特點是返回的函數(shù)還引用了外層函數(shù)的局部變量,所以,要正確使用閉包,就要確保引用的局部變量在函數(shù)返回后不能變。舉例如下:
# 希望一次返回3個函數(shù),分別計算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()
你可能認為調(diào)用f1(),f2()和f3()結(jié)果應(yīng)該是1,4,9,但實際結(jié)果全部都是 9(請自己動手驗證)。
原因就是當(dāng)count()函數(shù)返回了3個函數(shù)時,這3個函數(shù)所引用的變量 i 的值已經(jīng)變成了3。由于f1、f2、f3并沒有被調(diào)用,所以,此時他們并未計算 i*i,當(dāng) f1 被調(diào)用時:
?>>> f1() 9 # 因為f1現(xiàn)在才計算i*i,但現(xiàn)在i的值已經(jīng)變?yōu)?
因此,返回函數(shù)不要引用任何循環(huán)變量,或者后續(xù)會發(fā)生變化的變量。
返回閉包不能引用循環(huán)變量,請改寫count()函數(shù),讓它正確返回能計算1x1、2x2、3x3的函數(shù)。
參考答案:
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())
請驗證,完成請求
由于請求次數(shù)過多,請先驗證,完成再次請求
打開微信掃碼自動綁定
綁定后可得到
使用 Ctrl+D 可將課程添加到書簽
舉報