也叫名稱空間python
全局名稱空間:建立的存儲「變量名與值的關係」的空間叫作全局名稱空間閉包
局部名稱空間:在函數的運行中開闢的臨時的空間叫作局部名稱空間函數
內置名稱空間:內置名稱空間中存放了python解釋器爲咱們提供的名字:input,print,str,list,tuple...它們都是咱們熟悉的,拿過來就能夠用的方法。ui
python中的做用域分4種狀況:spa
- L:local,局部做用域,即函數中定義的變量;
- E:enclosing,嵌套的父級函數的局部做用域,即包含此函數的上級函數的局部做用域,但不是全局的(閉包常見);
- G:globa,全局變量,就是模塊級別定義的變量;
- B:built-in,系統固定模塊裏面的變量,好比int, bytearray等。
加載變量的優先級順序依次是:py 內置做用域>當前模塊中的全局(文件從上而下讀取)>外層做用域>局部做用域code
搜索變量的優先級順序依次是:做用域局部>外層做用域>當前模塊中的全局>python內置做用域,也就是LEGB。blog
固然,local 和 enclosing 是相對的,enclosing 變量相對上層來講也是 local 。內存
x = int(2.9) # int built-in g_count = 0 # global def outer(): o_count = 1 # enclosing def inner(): i_count = 2 # local print(o_count) print(i_count) # NameError: name 'i_count' is not defined inner() outer() # 正常打印:1 print(o_count) # NameError: name 'o_count' is not defined ''' 解釋: inner 內部 print(o_count): inner 內部沒有 o_count 變量,會去上一級(outer 內部)找,找到 outer 內部 print(i_count): 雖然 i_count 在 inner內部,但並不會去 inner 內部做用域找,而是找本身 outer 做用域,找不到就一級一級往上 全局 print(o_count): 同理 outer 內部 print(i_count) '''
在Python中,只有模塊(module),類(class)以及函數(def、lambda)纔會引入新的做用域,其它的代碼塊(如if、try、for等)是不會引入新的做用域的,以下代碼:作用域
if 2>1: x = 1 print(x) # 1
這個是沒有問題的,if並無引入一個新的做用域,x仍處在當前做用域中,後面代碼可使用。input
def test(): x = 2 print(x) # NameError: name 'x2' is not defined
x=6 def f2(): print(x) x=5 f2() # 變量是先聲明,再引用的 # 錯誤的緣由在於 print(x),解釋器會在局部做用域找,會找到x=5(函數已經加載到內存),但x使用在聲明前了,因此報錯: # local variable 'x' referenced before assignment.如何證實找到了x=5呢?簡單:註釋掉x=5,x=6 # 報錯爲:name 'x' is not defined #同理 x=6 def f2(): x+=1 #local variable 'x' referenced before assignment. x 使用以前已經被聲明瞭 #x+=1:x = x + 1;x 已經被聲明瞭,x=6,這裏等於 6 = 6 + 1,發生報錯 f2()
要修改:
x=6 def f2(): global x # 默認找 local 裏的 x,加上 global關鍵字讓他去找外面 global 的 x print(x) x=5 # 對 global 的 x 進行修改 f2() # 6 print(x) # 5
當內部做用域想修改外部做用域的變量時,就要用到global和nonlocal關鍵字了,當修改的變量是在全局做用域(global做用域)上的,就要使用global先聲明一下,代碼以下:
count = 10 def outer(): global count print(count) count = 100 print(count) outer() #10 #100
# global 能少用就少用,由於會對全局變量作出修改,影響全局其餘地方用這個全局變量
global關鍵字聲明的變量必須在全局做用域上,不能嵌套做用域上,當要修改嵌套做用域(enclosing做用域,外層非全局做用域)中的變量怎麼辦呢,這時就須要nonlocal關鍵字了
count = 200 def outer(): count = 10 # enclosing 嵌套做用域 def inner(): nonlocal count # 引用 enclosing 的 count = 10,若是不加,print(count) 會引用 enclosing 的 count = 10 #global count # 這裏引用最外層的 global 的 count = 200 count = 20 # 修改 enclosing 的 count print(count) inner() print(count) # 這裏的值不是 outer 的 count ,而是 inner 的 count outer() #20 #20
(1)變量查找順序:LEGB,局部做用域>外層做用域>當前模塊中的全局>python內置做用域;
(2)只有模塊、類、及函數才能引入新做用域;
(3)對於一個變量,內部做用域先聲明就會覆蓋外部變量,不聲明直接使用,就會使用外部做用域的變量;
(4)內部做用域要修改外部做用域變量的值時,全局變量要使用global關鍵字,嵌套做用域變量要使用nonlocal關鍵字。nonlocal是python3新增的關鍵字,有了這個關鍵字,就能完美的實現閉包了。閉包跟裝飾器有關係,在裝飾器裏介紹。