命名空間是名稱與對象之間的關(guān)系,可以將命名空間看做是字典,其中的鍵是名稱,值是對象。
命名空間不共享名稱。
在命名空間中的名稱能將任何python對象作為值,在不同的命名空間中相同的名稱可以與不同的對象相關(guān)聯(lián)。但是,如果存在名稱解析協(xié)議,則多個(gè)命名空間可以一起工作來解析名稱。也就是說,如果有多個(gè)命名空間(總是有的),那么可以定義搜索的順序,依次在不同的命名空間里來查找某個(gè)名稱(或確認(rèn)其不存在于任何認(rèn)可的命名空間)。在python中,將這一過程定義為作用域。
作用域搜索規(guī)則:LEGB
L:局部的(local)
E:封閉的(Enclosing)
G:全局的(Global)
B:內(nèi)置的(Built-in)
一、局部命名空間
函數(shù)內(nèi)部的命名空間,在調(diào)用函數(shù)的時(shí)候生成,調(diào)用結(jié)束時(shí)消失。當(dāng)局部命名空間有效時(shí),它是第一個(gè)用于檢查某個(gè)名字存在性的命名空間。如果在局部命名空間內(nèi)找到該名稱,則返回與名字相關(guān)聯(lián)的對象,反之提示出錯(cuò)。
二、全局命名空間
python在模塊中維護(hù)命名空間,模塊是一些python文件--包含函數(shù)等對象,并且可以導(dǎo)入其他程序使用。當(dāng)某個(gè)模塊被導(dǎo)入之后,該模塊同時(shí)引入了一個(gè)命名空間,其中包含模塊中所有的名稱和關(guān)聯(lián)的對象,可以通過存儲(chǔ)在沒個(gè)模塊中的__dict__來查看這個(gè)命名空間,換句話說,字典就是這個(gè)模塊的命名空間。
如果想要引用給模塊中的對象,要使用點(diǎn)符號(hào)將名稱和模塊名稱關(guān)聯(lián),這實(shí)際上是要求將對象與該模塊中的名稱相關(guān)聯(lián)。
當(dāng)python啟動(dòng)解釋器時(shí),它將自動(dòng)導(dǎo)入兩個(gè)模塊,即模塊__main__和__built-ins__。__main__模塊是默認(rèn)的全局模塊,所有新對象都存儲(chǔ)在其中??梢酝ㄟ^函數(shù)globals來訪問該命名空間的字典。子啊平python解釋器中通過輸入用戶交互時(shí),globals是有效的命名空間。
1.局部賦值規(guī)則
python中有一種稱為“本地賦值”的規(guī)則非常有趣。如果在函數(shù)內(nèi)的任何地方進(jìn)行局部賦值,則該賦值只在當(dāng)前活動(dòng)的命名空間中創(chuàng)建名稱。有時(shí)這將產(chǎn)生副作用,舉例如下:
>>>value=27
>>>deffunc(param1,param2):
forkey,valinlocals().items():
print(key,val)
value=value+1
>>>func(98765,43210)
param198765
param243210
Traceback(mostrecentcalllast):
File"",line1,in
func(98765,43210)
File"",line4,infunc
value=value+1
UnboundLocalError:localvariable'value'referencedbeforeassignment
>>>
首先通過賦值在全局命名空間中創(chuàng)建了變量value。也許你會(huì)認(rèn)為,當(dāng)函數(shù)值加1是會(huì)先在局部的命名空間中查找變量,無法找到時(shí)在全局命名空間中找到該名字??梢圆⒉皇沁@樣。
python提出如下假設(shè),如果在函數(shù)體內(nèi)的任何地方對變量賦值,則python將名稱添加到局部命名空間中。語句value=value+1對對象value進(jìn)行賦值。python假設(shè)無論在何處發(fā)生賦值,value都是函數(shù)func局部命名空間的一部分。當(dāng)python嘗試把1跟value相加時(shí),該value名稱在局部命名空間中,但它沒有關(guān)聯(lián)值,所以python報(bào)錯(cuò)。
問題在于python何時(shí)決定使value出現(xiàn)在局部命名空間中。實(shí)際value出現(xiàn)在局部命名空間中發(fā)生在代碼運(yùn)行前,即,在python運(yùn)行到函數(shù)定義之前。由于創(chuàng)建命名空間時(shí),python會(huì)檢查代碼并填充局部命名空間。在python運(yùn)行那行代碼之前,就發(fā)現(xiàn)了對value的賦值,并把它添加到局部命名空間中,當(dāng)函數(shù)執(zhí)行時(shí),python解釋器認(rèn)為value在局部命名空間中但沒有值,所以會(huì)產(chǎn)生錯(cuò)誤。
2.global語句
有一個(gè)方法可以解決上面的問題。如果在函數(shù)體內(nèi),使用global語句將變量聲明為全局變量,那么python不會(huì)為該變量在命名空間中創(chuàng)建局部名稱。
三、內(nèi)置模塊
遵循LEGB搜索規(guī)則,如果python不能在局部命名空間中找到某個(gè)名稱,則會(huì)在全局命名空間中繼續(xù)尋找,它尋找到的將是python的內(nèi)置名稱。
built-in模塊和其他模塊一樣,都具有__dict__屬性,這就是模塊的命名空間
四、封閉式變量
“封閉式”的作用域規(guī)則適應(yīng)于函數(shù)定義函數(shù)時(shí),也就是說,在函數(shù)體內(nèi)定義了一個(gè)新的函數(shù)。這個(gè)函數(shù)體內(nèi)的函數(shù)是外函數(shù)的局部命名空間中的一部分,意味著只有在外函數(shù)執(zhí)行期間才能夠運(yùn)行。完整的LEGB規(guī)則是先檢查局部命名空間,之后是封閉在局部命名空間中的其他函數(shù),之后是全局命名空間,在最后以內(nèi)置命名空間結(jié)束。
以上內(nèi)容為大家介紹了python命名空間與作用域,希望對大家有所幫助,如果想要了解更多Python相關(guān)知識(shí),請關(guān)注IT培訓(xùn)機(jī)構(gòu):千鋒教育。http://www.em-kal.com/