詳解Python循環(huán)作用域與閉包
前言
首先來看一段代碼
x_list = [i for i in range(30)] y_list = [i for i in range(10, 20)] for y in y_list: x_list = filter(lambda a: a != y, x_list) x_list = list(x_list) print(x_list) print(len(x_list))
這段代碼會輸出什么呢?
正確答案是一個長度為29的List。
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29]
29
但是實際上,上述代碼我們想要表達的意圖是從x_list中剔除所有在y_list中的元素。為什么在實際情況下,最終只會剔除一個元素呢?這主要與Python的作用域機制有關。
Python作用域機制
Python與其他語言不同,Python沒有循環(huán)作用域這個說法。Python的作用域遵循LEGB原則
- L, local – 在lambda函數(shù)內或者def函數(shù)內部的變量
- E, Enclosing-function – 閉包的作用域
- G,Global – 全局作用域
- B, Build-in – 內建作用域
為了證明Python沒有循環(huán)作用域,可以通過下面一段代碼驗證
for i in range(10): pass print(i)
運行代碼,發(fā)現(xiàn)可以正常運行,運行結果i==9。由此可以證明Python不存在循環(huán)作用域,循環(huán)變量屬于全局作用域。
基于上述結論,就可以很好地說明為什么上述的filter函數(shù)最終只去掉了一個元素。
因為filter函數(shù)是一個惰性函數(shù),因此在循環(huán)過程中并不會進行實際運算,而當循環(huán)完成,需要實際輸出的時候,此時全局作用域環(huán)境下的i已經變?yōu)榱艘粋€固定值19,因此最終只有19可以從x_list中去掉。
解決方案——閉包
面對上述問題,我們有兩個解決方案。
第一個解決方案——避免惰性求值??梢园l(fā)現(xiàn),問題的根源在于filter函數(shù)是一個惰性求值函數(shù),因此造成了這個問題??梢酝ㄟ^強制求值運算,強制每一次循環(huán)都進行filter操作,從而實現(xiàn)正常的篩選操作。代碼如下所示。
x_list = [i for i in range(30)] y_list = [i for i in range(10, 20)] for y in y_list: x_list = list(filter(lambda a: a != y, x_list)) x_list = list(x_list) print(x_list) print(len(x_list))
第二個解決方案——閉包。有時候我們不想放棄惰性求值這個特性,那么我們就需要引入更高級的函數(shù)式編程思想——閉包。
因為Python支持函數(shù)式編程語法,可以將函數(shù)作為變量,因此可以很容易的實現(xiàn)閉包特性。
x_list = [i for i in range(30)]
y_list = [i for i in range(10, 20)]
def check(a, b):
print('check')
return a != b
for y in y_list:
def x_filter(y):
global x_list
x_list = filter(lambda x: check(x, y), x_list)
x_filter(y)
print('loop')
x_list = list(x_list)
print(x_list)
print(len(x_list))
上面的代碼為了證明惰性求值的有效性,因此稍微繁瑣了一些。在實際場景中,check函數(shù)可以直接寫成lambda函數(shù)的形式。
閉包之所以能解決循環(huán)作用域問題,是因為閉包有獨立的作用域。因此即便是惰性求值,但是由于閉包作用于已經將臨時變量進行了存儲,因此依然可以正確進行篩選操作。
總結
Python與其他編程語言不同,不存在循環(huán)臨時作用域,因此在某些場景下會出現(xiàn)與其它編程語言結果不一致的BUG。面對這種情況,我們一般可以通過兩種方式來解決
1.避免惰性求值
2.使用閉包來保存循環(huán)臨時變量
以上所述是小編給大家介紹的Python循環(huán)作用域與閉包詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網站的支持!
相關文章
PyHacker編寫指南引用Nmap模塊實現(xiàn)端口掃描器
這篇文章主要為大家介紹了PyHacker編寫指南Nmap模塊實現(xiàn)端口掃描,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-05-05
Python?Pandas讀取Excel日期數(shù)據的異常處理方法
Excel文件是傳統(tǒng)的數(shù)據格式,但面對海量數(shù)據時,用編程的方法來處理數(shù)據更有優(yōu)勢,下面這篇文章主要給大家介紹了關于Python?Pandas讀取Excel日期數(shù)據的異常處理方法,需要的朋友可以參考下2022-02-02
Python3實現(xiàn)將本地JSON大數(shù)據文件寫入MySQL數(shù)據庫的方法
這篇文章主要介紹了Python3實現(xiàn)將本地JSON大數(shù)據文件寫入MySQL數(shù)據庫的方法,涉及Python針對json大數(shù)據文件的逐行讀取、mysql數(shù)據庫寫入等相關操作技巧,需要的朋友可以參考下2018-06-06

