詳解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的作用域機制有關(guān)。
Python作用域機制
Python與其他語言不同,Python沒有循環(huán)作用域這個說法。Python的作用域遵循LEGB原則
- L, local – 在lambda函數(shù)內(nèi)或者def函數(shù)內(nèi)部的變量
- E, Enclosing-function – 閉包的作用域
- G,Global – 全局作用域
- B, Build-in – 內(nèi)建作用域
為了證明Python沒有循環(huán)作用域,可以通過下面一段代碼驗證
for i in range(10): pass print(i)
運行代碼,發(fā)現(xiàn)可以正常運行,運行結(jié)果i==9。由此可以證明Python不存在循環(huán)作用域,循環(huán)變量屬于全局作用域。
基于上述結(jié)論,就可以很好地說明為什么上述的filter函數(shù)最終只去掉了一個元素。
因為filter函數(shù)是一個惰性函數(shù),因此在循環(huán)過程中并不會進行實際運算,而當(dāng)循環(huán)完成,需要實際輸出的時候,此時全局作用域環(huán)境下的i已經(jīng)變?yōu)榱艘粋€固定值19,因此最終只有19可以從x_list中去掉。
解決方案——閉包
面對上述問題,我們有兩個解決方案。
第一個解決方案——避免惰性求值。可以發(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)作用域問題,是因為閉包有獨立的作用域。因此即便是惰性求值,但是由于閉包作用于已經(jīng)將臨時變量進行了存儲,因此依然可以正確進行篩選操作。
總結(jié)
Python與其他編程語言不同,不存在循環(huán)臨時作用域,因此在某些場景下會出現(xiàn)與其它編程語言結(jié)果不一致的BUG。面對這種情況,我們一般可以通過兩種方式來解決
1.避免惰性求值
2.使用閉包來保存循環(huán)臨時變量
以上所述是小編給大家介紹的Python循環(huán)作用域與閉包詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
pycharm配置pyqt5-tools開發(fā)環(huán)境的方法步驟
這篇文章主要介紹了pycharm配置pyqt5-tools開發(fā)環(huán)境的方法步驟,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-02-02如何修復(fù)使用 Python ORM 工具 SQLAlchemy 時的常見陷阱
SQLAlchemy 是一個 Python ORM 工具包,它提供使用 Python 訪問 SQL 數(shù)據(jù)庫的功能。這篇文章主要介紹了如何修復(fù)使用 Python ORM 工具 SQLAlchemy 時的常見陷阱,需要的朋友可以參考下2019-11-11python實現(xiàn)讀取excel寫入mysql的小工具詳解
EXCEL 和 MySQL 大體上來說都可以算是"數(shù)據(jù)庫",MySQL貌似有EXCEL的接口,但是最近在自學(xué)Python,用Python實現(xiàn)了一下,下面這篇文章主要給大家介紹了關(guān)于利用python實現(xiàn)讀取excel寫入mysql的一個小工具,需要的朋友可以參考下。2017-11-11Python下實現(xiàn)的RSA加密/解密及簽名/驗證功能示例
這篇文章主要介紹了Python下實現(xiàn)的RSA加密/解密及簽名/驗證功能,結(jié)合具體實例形式分析了Python中RSA加密、解密的實現(xiàn)方法及簽名、驗證功能的使用技巧,需要的朋友可以參考下2017-07-07在Python的Django框架中用流響應(yīng)生成CSV文件的教程
這篇文章主要介紹了在Python的Django框架中用流響應(yīng)生成CSV文件的教程,作者特別講到了防止CSV文件中的中文避免出現(xiàn)亂碼等問題,需要的朋友可以參考下2015-05-05