Python延遲綁定問題原理及解決方案
延遲綁定出現(xiàn)在閉包問題中。下面我們看一個閉包的例子:
def (n): def mul(x): return n*x return mul double = gen_mul(2) doubled_value = double(6)
可以看出滿足閉包的幾點:
- 有內(nèi)部函數(shù)
- 內(nèi)部函數(shù)引用了外部函數(shù)中的自由變量
- 內(nèi)部函數(shù)被返回
閉包的優(yōu)點:
- 可以避免使用全局變量
- 可以持久化變量,達到靜態(tài)變量的作用
閉包的缺點:
- 可能會消耗大量的內(nèi)存
- 可能會導(dǎo)致內(nèi)存泄漏
當然缺點可以通過人為避免。
現(xiàn)在我們來看看另一個會引出延遲綁定的例子:
def multipliers(): return [lambda x : i * x for i in range(4)] print([m(2) for m in multipliers()]) # [6,6,6,6]
上邊的例子會輸出[6,6,6,6],而不是我們預(yù)期的[0,2,4,6]。
這就是延遲綁定導(dǎo)致的結(jié)果。具體過程我們可以來分析下:
執(zhí)行第三行時,會先執(zhí)行multipliers函數(shù),然后執(zhí)行函數(shù)中的列表解析式。在每一次迭代的時候都會生成一個匿名函數(shù)(這里只是定義)作為元素。然后回到第三行,遍歷返回的列表中的匿名函數(shù),傳入?yún)?shù)2并執(zhí)行。此時函數(shù)類似于這樣:
def noname(x):
return i * x
我們知道Python查找變量的作用域鏈的順序依次為LEGB:
局部變量(L)->外部函數(shù)中的局部變量(E)->全局變量(G)->內(nèi)置變量(B)
非常重要的一點我們需要知道:Python的作用域在編譯時就已經(jīng)形成了,而不是在運行時,函數(shù)的作用域與其被調(diào)用的位置無關(guān)。
那么在本例中,上面的noname函數(shù)體中的i從何而來呢?當然首先會到multipliers函數(shù)的局部變量中去尋找。此時i的值已經(jīng)為3,所以出現(xiàn)這種讓人”費解”的現(xiàn)象。
那么現(xiàn)在我們既然已經(jīng)知道了原因,那么要怎樣解決呢?
我們可以將迭代的i值直接注入到匿名函數(shù)的函數(shù)體中,這里給出兩種方法:
通過為參數(shù)設(shè)置默認值,這是因為在編譯時就會計算確定默認值:
def multipliers_ch1():
return [lambda m,x=i : m * x for i in range(4)]
通過內(nèi)置函數(shù)partial:
from functools import partial def multipliers_ch2(): return [partial(lambda m,x : m * x,i) for i in range(4)]
利用生成器的延遲計算:
def multipliers_ch3(): for m in range(4): yield lambda x: m * x
partial及生成器的內(nèi)容會在以后分享。
運行結(jié)果
print([m(2) for m in multipliers_ch1()]) # [0,2,4,6]
print([m(2) for m in multipliers_ch2()]) # [0,2,4,6]
print([m(2) for m in multipliers_ch3()]) # [0,2,4,6]
注:
自由變量:指未在本地作用域中綁定的變量,我們可通過訪問函數(shù)的code屬性進行查看:
fun.code.co_freevars
LEGB: 可看該部分解釋
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
利用Pandas讀取文件路徑或文件名稱包含中文的csv文件方法
今天小編就為大家分享一篇利用Pandas讀取文件路徑或文件名稱包含中文的csv文件方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-07-07python concurrent.futures模塊的使用測試
大家都知道concurrent.futures 是 3.2 中引入的新模塊,它為異步執(zhí)行可調(diào)用對象提供了高層接口,今天通過本文給大家介紹python concurrent.futures模塊的使用測試 ,感興趣的朋友一起看看吧2021-07-07django配置連接數(shù)據(jù)庫及原生sql語句的使用方法
這篇文章主要給大家介紹了關(guān)于django配置連接數(shù)據(jù)庫,以及原生sql語句的使用方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03Pandas DataFrame操作數(shù)據(jù)增刪查改
我們在用 pandas 處理數(shù)據(jù)的時候,經(jīng)常會遇到用其中一列數(shù)據(jù)替換另一列數(shù)據(jù)的場景。這一類的需求估計很多人都遇到,當然還有其它更復(fù)雜的。解決這類需求的辦法有很多,這里我們來推薦幾個,這篇文章主要介紹了Pandas DataFrame操作數(shù)據(jù)的增刪查改2022-10-10