命名空間
一個程序具有多個變量,為了更好的組織這些變量,更好的控制變量之間的命名沖突以及更好的管理變量在程序中的角色,python使用了命名空間這樣一種結(jié)構(gòu)來實現(xiàn)這個目的。python中,一個命名空間由多個變量組成,且以字典的形式來保存這些變量,其中key是變量名稱,value是變量名對應(yīng)的對象。
同一個命名空間中的變量名是唯一的,不同命名空間中的變量相互獨立(這里的相互獨立指的是不會相互沖突相互覆蓋,但是可能會有相同的對象引用(即指向同一個內(nèi)存中的對象))。既然命名空間是為了更好的控制變量之間的命名沖突以及更好的管理變量在程序中的角色,那么什么情況下把兩個變量放入同一個命名空間,即以什么樣的邏輯和規(guī)則定義變量的命名空間就是最基本的問題。
python中約定了四種命名空間:內(nèi)置的(Bulit-in)、全局的(Global)、外層函數(shù)的(Enclosing or Nonlocal)以及本地的(Local)。
這四種命名空間的定義規(guī)則和邏輯如下:
內(nèi)置的(Bulit-in):python定義的所有的內(nèi)置變量,比如abs、max這些內(nèi)置函數(shù)以及預(yù)定義的異常等,這些變量都放在內(nèi)置命名空間中,可以通過python的標準模塊builtins獲取所有的內(nèi)置變量。
全局的(Global):在模塊(腳本)頂層定義的所有變量,python把這些變量會放在該模塊的全局命名空間中,可以通過globals()獲取當前位置已經(jīng)定義了的所有全局變量。
本地的(Local):python中函數(shù)的每次調(diào)用會新創(chuàng)建一個自己的命名空間,這個命名空間包含的變量為函數(shù)參數(shù)和函數(shù)中定義的所有變量,這些變量也成為該函數(shù)的本地變量。
外層函數(shù)的(Enclosing or Nonlocal):如果函數(shù)A中嵌套了一個函數(shù)B,那么對于內(nèi)層函數(shù)B來說,外層函數(shù)A的本地命名空間就是B的外層函數(shù)命名空間。單純從命名空間的角度看,似乎外層函數(shù)命名空間和本地命名空間重復(fù)定義了,因為其本身就是外層函數(shù)的本地命名空間。但從作用域和變量查找的角度看,并不會重復(fù)定義;實際上,在python2.2之前,這層命名空間是不存在的,2.2版本的python才加入了這層命名空間,加入后,使得函數(shù)B直接引用函數(shù)A的本地變量變?yōu)榭赡?因此,對于這種嵌套函數(shù)形式下的外層函數(shù)命名空間,python給了其一個單獨的定義。
作用域和LEGB變量查找規(guī)則
定義好命名空間之后,我們知道具有同一個名稱的不同變量可能同時存在于多個不同的命名空間中,那么問題來了:當我們引用一個變量x,x同時存在于多個命名空間中,python如何知道我們引用的是哪一個?python對此引入了作用域和變量查找規(guī)則來消除這種歧義性。
域( scope)指的就是程序的文本區(qū)域。python中的域指的就是變量的作用域,某個變量的作用域指的是這樣一個區(qū)域在這個區(qū)域中,python可以直接獲取到該變量,直接獲取的意思是可以直接在該變量的命名空間中找到該變量,而不是通過屬性訪問等其他方式獲取。 所以通過明確變量的作用域可以讓我們知道某個變量在程序的哪些地方起作用(可以被直接獲取),同時約定了變量作用域還可以消除部分的歧義性,因為對于某個變量并不起作用的區(qū)域,我們自然就不需要在該區(qū)域考慮該變量。
對于上述四種不同命名空間里的變量,其各自有自己的作用域,分別如下:
全局(變量)作用域:指的就是全局變量的作用域,也就是整個模塊文件;
內(nèi)置(變量)作用域:指的就是內(nèi)置變量的作用域,包括整個解釋器環(huán)境;
外層函數(shù)(變量)作用域:也指非本地作用域,指的就是外層函數(shù)中變量的作用域,包括整個外層函數(shù)下的代碼塊,同時也是外層函數(shù)的本地作用域;
本地(變量)作用域:就是一個函數(shù)下定義的本地變量的作用域,為函數(shù)下的代碼塊區(qū)域。
明確上述四種變量的各自作用域后,依然存在歧義性,因為不同命名空間中的相同名稱的變量的作用域可能存在重復(fù),比如有一個全局變量x,在一個函數(shù)中又定義了一個本地變量x,那么該函數(shù)中的代碼塊即是全局變量x的作用域,也是本地變量x的作用域。針對這種情況,python中規(guī)定了變量在不同命名空間中的查找順序,依次為:本地命名空間(L)、外層函數(shù)命名空間(E or N)、全局命名空間(G)以及內(nèi)置命名空間(B),這稱為python變量查找的LEGB(LNGB)規(guī)則。如此,即使不同命名空間中存在相同名稱的變量且作用域重復(fù),根據(jù)該規(guī)則,就可以確定python查找的變量對應(yīng)的值到底是哪一個,從而可以消除變量名的歧義性。
需要注意的是,LEGB(LNGB)規(guī)則是python將源碼編譯成字節(jié)碼時的變量作用域解析規(guī)則,所以實際上,變量的命名空間在編譯時就被靜態(tài)確定了,從而在解釋器執(zhí)行編譯后的字節(jié)碼時,python會直接在變量對應(yīng)的命名空間中去獲取該對象。
最后強調(diào)一下外層函數(shù)命名空間和作用域,該命名空間是在python2.2版本加入的,在2.2之前的版本,如下代碼如無法運行成功,因為對于inner函數(shù)中的x,python只會在本地命名空間、全局命名空間和內(nèi)置命名空間中查找,outer函數(shù)的命名空間會被直接跳過,因此下面的代碼會因為找不到x而報錯。2.2之前的版本為了使用outer函數(shù)的x,一般需要把x作為參數(shù)傳給inner。但是在2.2版本開始,加入了Enclsoing function概念,也就是outer函數(shù),該函數(shù)的命名空間為外層函數(shù)命名空間,其變量的作用域包含了inner函數(shù)的代碼塊,因此python在查找x變量時,也會查找該外層函數(shù)的命名空間。