本篇文章的技術方向,主要是給大家解讀Python中閉包與lambda的實現方法,對以后的工作中,具有一定的參考價值,希望可以為各位讀者帶來幫助。
Python閉包與lambda的作用域
lambda寫法
def fun():
for i in range(3):
yield lambda x : x * i
f0, f1, f2 = fun()
print(f0(1), f1(2), f2(3))
閉包的寫法
def fun():
result = []
for i in range(3):
def demo(x):
return x * i
result.append(demo)
return result
f0, f1, f2 = fun()
print(f0(1), f1(2), f2(3))
上面兩種寫法的結果都是2,4,6,按最初的想法結果應該是0,2,6。
問題原因:
問題的本質原因,主要是利用python的相關變量,去進行查找相關的規則,在以上舉例子中,i指的是在閉包中的相關作用域,而在python中,其作用是截然相反的,而在閉包當中的話,其變量i的值,指的意思是,如果我們在使用內部函數調用時,就容易被查詢到
解決辦法
有一個比較好的解決方法,就是將閉包應用范圍變為局部應用范圍
lambda寫法
def fun():
for i in range(3):
yield lambda x, i = i: x * i
f0, f1, f2 = fun()
print(f0(1), f1(2), f2(3))
閉包的寫法
def fun():
result = []
for i in range(3):
def demo(x):
return x * i
result.append(demo)
return result
f0, f1, f2 = fun()
print(f0(1), f1(2), f2(3))
以上輸出結果,0,2,6。
另一種情況:
def fun():
for i in range(3):
yield lambda x : x * i
f0, f1, f2 = fun()
print(f0(1), f1(2), f2(3))
輸出結果還是2,4,6
問題原因
fun()方法返回的)變成了應用功能器,并不是實際變成的,只有我們在用到應用功能器的時候,才會去進行一些相關的調用。
在遍歷后執行打印時,i的值,是最后才能夠進行調用的值。將lambda看作閉包方法的話,變量i值還是閉包作用域(no local)
python當中的坑(閉包與lambda)
先來看一個例子
def create():
return [lambda x:i*x for i in range(5)]
for i in create():
print(i(2))
結果:
8
8
8
8
8
create函數的返回值時一個列表,列表的每一個元素都是一個函數--將輸入參數x乘以一個倍數i的函數。預期的結果時0,2,4,6,8.但結果是5個8,意外不意外。
出現這個問題的原因是使用了lambda,所以很多人認為造成這種現象的原因是lambda的問題,但是我們需要做的是,要仔細的對其進行簡單。問題的本質在與python中的屬性查找規則,LEGB(local,enclousing,global,bulitin),在上面的例子中,i就是在閉包作用域(enclousing),而Python的閉包是遲綁定,這意味著閉包中用到的變量的值,是在內部函數被調用時查詢得到的
解決辦法也很簡單,那就是變閉包作用域為局部作用域。
def create():
return [lambda x, i=i:i*x for i in range(5)]
for i in create():
print(i(2))
換種寫法:
def create():
a = []
for i in range(5):
def demo(x, i=i):
return x*i
a.append(demo)
return a
for i in create():
print(i(2))
以上兩種寫法是一樣的
結果:
0
2
4
6
8
下面我給大家再舉一個相似的例子:
nums = range(2,20)
for i in nums:
nums = filter(lambda x: x==i or x%i, nums)
print(list(nums))
結果:
[2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19]
同樣按照正常的邏輯結果應該為:
[2, 3, 5, 7, 11, 13, 17, 19]
問題產生的原因:
·
在python3當中filter()函數返回是一個進行迭代的機器,它的執行并不是完整的進行執行的,而是在每次調用的時候執行(python2中filter()返回的值列表,無此現象)
·
·
在遍歷后執行打印時,現在執行循環當中的函數,如果上面有一個同樣的問題,i這個變量值和最后調用的一個值是一樣的,與以上例子不同的是,以上例子用的是內嵌作用域的值,而這個例子用的是全局i的值。
·
修改代碼:
nums = range(2,20)
for i in nums:
nums = filter(lambda x,i=i: x==i or x%i, nums)
print(list(nums))
結果:
[2, 3, 5, 7, 11, 13, 17, 19]
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/127597.html
摘要:恩如期來啦閉包一函數作為返回值介紹閉包之前,先了解一下函數作為返回值的情況。例如之前介紹的裝飾器中,就出現了將函數作為返回值。當執行時,相當于執行,且包含。允許使用關鍵字創造匿名函數。例如調用默認可以把匿名函數作為返回值返回,例如 恩~ 如期來啦閉包~ 一、函數作為返回值 介紹閉包之前,先了解一下函數作為返回值的情況。高階函數除了可以接收函數作為參數外,還可以把函數作為結果值返回。...
摘要:根據我們已知的運行模式,我們沒法在函數返回之后繼續調用函數,在函數被調用的時候,變量早已不復存在,可能會發生一個運行時錯誤。萬萬沒想到,返回的函數居然能夠正常工作。 文件讀寫 在Python中,文件的操作應用非常多,比如大數據領域,涉及許多數據處理請求,基本上都是從一個文件對數據進行分析、抽取、重寫進行梳理數據 文件的讀寫分幾步?如何把大象放冰箱showImg(https://segm...
摘要:表達式又名閉包匿名函數筆記根據終于在中引入了表達式。函數式接口要介紹中表達式的實現,需要知道什么是函數式接口。但同樣需要保證外部的自由變量不能在表達式中被改變。 Java Lambda 表達式(又名閉包 (Closure)/ 匿名函數 ) 筆記 根據 JSR 335, Java 終于在 Java 8 中引入了 Lambda 表達式。也稱之為閉包或者匿名函數。 showImg(https...
摘要:上一講和本講的標題是大話題小函數,所謂大話題,就是這些函數如果溯源,都會找到聽起來更高大上的東西。是提出的一個軟件架構,用于大規模數據集大于的并行運算。 上一講和本講的標題是大話題小函數,所謂大話題,就是這些函數如果溯源,都會找到聽起來更高大上的東西。這種思維方式絕對我堅定地繼承了中華民族的優良傳統的。自從天朝的臣民看到英國人開始踢足球,一直到現在所謂某國勃起了,都一直在試圖論證足球起...
摘要:函數返回值支持多個,返回多個時,使用逗號分隔。如上,與這個參數按照由左到右依次賦值給形參和供函數內部使用。在傳入函數時,便是將這個引用傳入了函數。需要使用關鍵字聲明將變量作用域變為函數外部非全局變量,及閉包作用域。 < 返回索引頁 函數 函數定義 返回值 函數的調用 函數的參數 順序參數 參數的默認值 可變參數 關鍵字參數 參數的組合 參數的專遞 函數作用域 內建...
關注公眾號:「Python專欄」,后臺回復「爬蟲書籍」,即可獲得2本Python爬蟲相關的電子書showImg(https://segmentfault.com/img/remote/1460000018448215?w=860&h=860); 無意間,看到這么一道Python面試題:以下代碼將輸出什么? def testFun(): temp = [lambda x : i*x for ...
閱讀 910·2023-01-14 11:38
閱讀 877·2023-01-14 11:04
閱讀 740·2023-01-14 10:48
閱讀 1980·2023-01-14 10:34
閱讀 941·2023-01-14 10:24
閱讀 818·2023-01-14 10:18
閱讀 498·2023-01-14 10:09
閱讀 571·2023-01-14 10:02