Python命名空間詳解
通俗的來(lái)說(shuō),Python中所謂的命名空間可以理解為一個(gè)容器。在這個(gè)容器中可以裝許多標(biāo)識(shí)符。不同容器中的同名的標(biāo)識(shí)符是不會(huì)相互沖突的。理解python的命名空間需要掌握三條規(guī)則:
第一,賦值(包括顯式賦值和隱式賦值)產(chǎn)生標(biāo)識(shí)符,賦值的地點(diǎn)決定標(biāo)識(shí)符所處的命名空間。
第二,函數(shù)定義(包括def和lambda)產(chǎn)生新的命名空間。
第三,python搜索一個(gè)標(biāo)識(shí)符的順序是"LEGB"。
所謂的"LEGB"是python中四層命名空間的英文名字首字母的縮寫(xiě)。
最里面的一層是L(local),表示在一個(gè)函數(shù)定義中,而且在這個(gè)函數(shù)里面沒(méi)有再包含函數(shù)的定義。
第二層E(enclosing function),表示在一個(gè)函數(shù)定義中,但這個(gè)函數(shù)里面還包含有函數(shù)的定義,其實(shí)L層和E層只是相對(duì)的。
第三層G(global),是指一個(gè)模塊的命名空間,也就是說(shuō)在一個(gè).py文件中定義的標(biāo)識(shí)符,但不在一個(gè)函數(shù)中。
第四層B(builtin),是指python解釋器啟動(dòng)時(shí)就已經(jīng)具有的命名空間,之所以叫builtin是因?yàn)樵趐ython解釋器啟動(dòng)時(shí)會(huì)自動(dòng)載入__builtin__模塊,這個(gè)模塊中的list、str等內(nèi)置函數(shù)的就處于B層的命名空間中。
這三條規(guī)則通過(guò)一個(gè)例子來(lái)看比較明白。如下面例子所示:
>>> g = int('0x3', 0) >>> def outFunc(): e = 2 g = 10 def inFunc(): l = 1 return g + e return inFunc() >>> outFunc() ===> 12
來(lái)詳細(xì)看看這段代碼中的標(biāo)識(shí)符。
第1行,適用第一條規(guī)則“賦值產(chǎn)生標(biāo)識(shí)符”,因此產(chǎn)生一個(gè)標(biāo)識(shí)符g?!百x值的地點(diǎn)決定標(biāo)識(shí)符所處的命名空間”,因?yàn)間是沒(méi)有在一個(gè)函數(shù)定義中,因此g處于'G'層命名空間中。這一行中還有一個(gè)標(biāo)識(shí)符,那就是int。那么int是在什么地方定義的呢?由于int是內(nèi)置函數(shù),是在__builtin__模塊中定義的,所以int就處于'B'的層命名空間中。
第2行,適用第一條規(guī)則,由于def中包含一個(gè)隱性的賦值過(guò)程,這一行產(chǎn)生一個(gè)標(biāo)識(shí)符outFunc,outFunc并不處于一個(gè)函數(shù)定義的內(nèi)部,因此,outFunc處于'G'層命名空間中。此外,這一行還適用第二條規(guī)則,產(chǎn)生一個(gè)新的命名空間。
第3行,適用第一條規(guī)則,產(chǎn)生個(gè)標(biāo)識(shí)符e,而且由于這是在一個(gè)函數(shù)定義內(nèi),并且內(nèi)部還有函數(shù)定義,因此e處于'E'層命名空間中。
第4行要注意,適用第一條規(guī)則,產(chǎn)生一個(gè)標(biāo)識(shí)符g,這個(gè)g與e一樣外于'E'層命名空間中。這個(gè)g與第一行的g是不同的,因?yàn)樗幍拿臻g不一樣。
第5行,適用第一條規(guī)則,產(chǎn)生一個(gè)處于'E'層命名空間的標(biāo)識(shí)符inFunc。與第2行一樣,這一行定義函數(shù)也產(chǎn)生一個(gè)新的命名空間。
第6行,適用第一條規(guī)則,產(chǎn)生一個(gè)標(biāo)識(shí)符l,由于這個(gè)l處于一個(gè)函數(shù)內(nèi)部,而且在這個(gè)函數(shù)內(nèi)部沒(méi)有其他函數(shù)的定義,因此l處于'L'層命名空間中。
第7行,適用第三條規(guī)則,python解釋器首先看到標(biāo)識(shí)符g,按照LEGB的順序往上找,先找L層(也就是在inFunc內(nèi)部),沒(méi)有。再找E層,有,值為10。因此這里的g的值為10。尋找過(guò)程到為止,并不會(huì)再往上找到'G'層。尋找e的過(guò)程也一樣,e的值為2。因此第9行的結(jié)果為12。
其實(shí),所謂的“LEGB”是為了學(xué)術(shù)上便于表述而創(chuàng)造的。讓一個(gè)編程的人說(shuō)出哪個(gè)標(biāo)識(shí)符處于哪個(gè)層沒(méi)有什么意義,只要知道對(duì)于一個(gè)標(biāo)識(shí)符,python是怎么尋找它的值的就可以了。其實(shí)找值的過(guò)程直觀上也很容易理解。
通過(guò)上面的例子也可以看出,如果在不同的命名空間中定義了相同的標(biāo)識(shí)符是沒(méi)有關(guān)系的,并不會(huì)產(chǎn)生沖突。尋找一個(gè)標(biāo)識(shí)符的值過(guò)程總是從當(dāng)前層開(kāi)始往上找的,首先找到的就為這個(gè)標(biāo)識(shí)符的值。也由此可以這么說(shuō),'B'層標(biāo)識(shí)符在所有模塊(.py文件)中可用;'G'層標(biāo)識(shí)符在當(dāng)前模塊內(nèi)(.py文件)中可用;'E'和'L'層標(biāo)識(shí)符在當(dāng)前函數(shù)內(nèi)可用。
再來(lái)看一個(gè)例子,來(lái)解釋global語(yǔ)句的用法。代碼如下所示:
>>> g = 'global' >>> s = 'in' >>> def out(): g = 'out' def inter(): global g print s,g inter() >>> out() ===> 'in global'
可以看到,雖然有兩個(gè)層中的g,但使用了global語(yǔ)句后,就是指'G'層的標(biāo)識(shí)符。也就是第7行中的g,就是指第1行產(chǎn)生的那個(gè)g,值為'global'。
最后說(shuō)一句,其實(shí)只要在編程的時(shí)候注意一下,不要使用相同的標(biāo)識(shí)符,基本上就可以避免任何與命名空間相關(guān)的問(wèn)題。還有就是在一個(gè)函數(shù)中盡量不要使用上層命名空間中的標(biāo)識(shí)符,如果一定要用,也最好使用參數(shù)傳遞的方式進(jìn)行,這樣有利于保持函數(shù)的獨(dú)立性。
相關(guān)文章
Python+matplotlib實(shí)現(xiàn)華麗的文本框演示代碼
這篇文章主要介紹了Python+matplotlib實(shí)現(xiàn)華麗的文本框演示代碼,分享了相關(guān)代碼示例,小編覺(jué)得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01Python地理地圖可視化folium標(biāo)記點(diǎn)彈窗設(shè)置代碼(推薦)
這篇文章主要介紹了Python地理地圖可視化folium標(biāo)記點(diǎn)彈窗設(shè)置,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-09-09python基于Bokeh庫(kù)制作子彈圖及瀑布圖示例教程
這篇文章主要為大家介紹了python基于Bokeh庫(kù)制作子彈圖及瀑布圖的實(shí)現(xiàn)示例教程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪2021-10-10pycharm遠(yuǎn)程調(diào)試openstack的圖文教程
這篇文章主要為大家詳細(xì)介紹了pycharm遠(yuǎn)程調(diào)試openstack的圖文教程,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11