在Python語言中,可以在函數(shù)中定義函數(shù)。 這種在函數(shù)中嵌套定義的函數(shù)也叫內部函數(shù)。我們來看下面的代碼:
創(chuàng)新互聯(lián)是一家專注于做網站、成都網站制作與策劃設計,柘城網站建設哪家好?創(chuàng)新互聯(lián)做網站,專注于網站建設10多年,網設計領域的專業(yè)建站公司;建站業(yè)務涵蓋:柘城等地區(qū)。柘城做網站價格咨詢:18980820575
上述代碼中,定義了函數(shù)greet,在函數(shù)greet內部又定義了一個函數(shù)inner_func, 并調用該函數(shù)打印了一串字符。
我們可以看到,內部函數(shù)inner_func的定義和使用與普通函數(shù)基本相同。需要注意的是變量的作用域,在上述代碼中,函數(shù)參數(shù)name對于全局函數(shù)greet是局部變量,對內部函數(shù)inner_func來說則是非局部變量。內部函數(shù)對于非局部變量的訪問規(guī)則類似于標準的外部函數(shù)訪問全局變量。
從這個例子我們還可以看到內部函數(shù)的一個作用,就是通過定義內部函數(shù)的方式將一些功能隱藏起來,防止外部直接調用。常見的場景是,在一個復雜邏輯的函數(shù)中,將一些小的任務定義成內部函數(shù),然后由這個外層函數(shù)使用,這樣可以使代碼更為清晰,易于維護。這些內部函數(shù)只會在這個外層函數(shù)中使用,不能被其他函數(shù)或模塊使用。
在Python語言中, 函數(shù)也是對象,它可以被創(chuàng)建、賦值給變量,或者作為函數(shù)的返回值。我們來看下面這個例子。
在上述代碼中,在函數(shù)gen_greet內部定義了inner_func函數(shù),并返回了一個inner_func函數(shù)對象。外部函數(shù)gen_greet返回了一個函數(shù)對象,所以像gen_greet這樣的函數(shù)也叫工廠函數(shù)。
在內部函數(shù)inner_func中,使用了外部函數(shù)的傳參greet_words(非局部變量),以及函數(shù)的參數(shù)name(局部變量),來打印一個字符串。
接下來,調用gen_greet("Hello")創(chuàng)建一個函數(shù)對象say_hello,緊接著調用say_hello("Mr. Zhang"),輸出的結果為:Hello, Mr. Zhang!
同樣的,調用gen_greet("Hi")創(chuàng)建一個函數(shù)對象say_hi,調用say_hello("Mr. Zhang"),輸出的結果為:Hi,Tony!
我們可以發(fā)現(xiàn),gen_greet返回的函數(shù)對象具有記憶功能,它能夠把所需使用的非局部變量保存下來,用于后續(xù)被調用的時候使用。這種保存了非局部變量的函數(shù)對象被稱作閉包(closure)。
那么閉包是如何實現(xiàn)的呢?其實并不復雜,函數(shù)對象中有一個屬性__closure__,它就是在創(chuàng)建函數(shù)對象時用來保存這些非局部變量的。
__closure__屬性是一個元組或者None類型。在上述代碼中,我們可以通過下面方式查看:
函數(shù)的嵌套所實現(xiàn)的功能大都可以通過定義類的方式來實現(xiàn),而且類是更加面向對象的代碼編寫方式。
嵌套函數(shù)的一個主要用途是實現(xiàn)函數(shù)的裝飾器。我們看下面的代碼:
在上述代碼中,logger函數(shù)返回函數(shù)with_logging,with_logging則是打印了函數(shù)func的名稱及傳入的參數(shù),然后調用func, 并將參數(shù)傳遞給func。其中的@wraps(func)語句用于復制函數(shù)func的名稱、注釋文檔、參數(shù)列表等等,使得with_logging函數(shù)具有被裝飾的函數(shù)func相同的屬性。
代碼中接下來用@logger對函數(shù)power_func進行修飾,它的作用等同于下面的代碼:
可見,裝飾器@符其實就是上述代碼的精簡寫法。
通過了解了嵌套函數(shù)和閉包的工作原理,我們在使用過程中就能夠更加得心應手了。
可以首先使用json包的loads函數(shù)對json數(shù)據進行解析,然后就可以像操作Python數(shù)據格式一樣對數(shù)據進行索引和遍歷了。
import json
s = '{"aescCityList":null,"cityAllList":null,"cityJsonArray"...'
data = json.loads(s)
for city in data["cityJsonArray"]:
if city["cityId"] == 4:
print city
該部分內容涉及 Python 變量作用域相關知識,變量作用域指的是變量的有效作用范圍,直接理解就是 Python 中的變量不是任意位置都可以訪問的,有限制條件。
一般情況下變量的作用域變化范圍是 塊級、函數(shù)、類、模塊、包等,級別是從小到達。Python 中是沒有塊級作用域的,所以我們在寫代碼的時候,下面的代碼是正確的。
在 Python 中常見的塊級作用域有 if 語句、for 語句、while 語句、with 上下文語句。
上文已經提及了作用域是 Python 程序可以直接訪問一個變量的作用范圍,Python 的作用域一共有 4 種,分別如下:
一個比較經典的案例如下:
在 Python 中變量尋找的順序是從內到外,先局部,然后外部,在全局,在內建,這種規(guī)則叫做 LEGB 規(guī)則 。
增加以下學習的趣味性,你可以研究下述代碼中變量是如何變化的。
定義在 函數(shù)內部 的變量擁有一個局部作用域,定義在 函數(shù)外部 的變量擁有全局作用域。
輸出結果,函數(shù)內部是 123 ,函數(shù)外部依舊是 0 。
如果希望函數(shù)內部(內部作用域)可以修改外部作用域的變量,需要使用 global 關鍵字。
此時輸出的就都是 123 了,還有一點需要注意,在函數(shù)內容如果希望修改全局變量的值, global 關鍵字一定要寫在變量操作前。
該代碼會出現(xiàn)語法錯誤:
全局變量還存在一個面試真題,經常出現(xiàn),請問下述代碼運行結果。
如果要修改嵌套作用域(Enclosing 作用域)中的變量,需要 nonlocal 關鍵字,測試代碼如下:
輸出結果自行測試,注意 nonlocal 關鍵字必須是 Python3.X+版本,Python 2.X 版本會出現(xiàn)語法錯誤:
在多重嵌套中, nonlocal 只會上溯一層,如果上一層沒有,則會繼續(xù)上溯,下述代碼你可以分別注釋查看結果。
局部變量和全局變量具體有哪些,可以通過 locals() 和 globals() 兩個內置函數(shù)獲取。
本篇博客為大家說明了 Python 的作用域,并且對 global 和 nonlocal 關鍵字進行了學習,希望對你有所幫助。