Python之函數做用域

一、做用域介紹

也叫名稱空間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
def、class、lambda是能夠引入新做用域的。


三、變量的修改

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關鍵字

當內部做用域想修改外部做用域的變量時,就要用到global和nonlocal關鍵字了,當修改的變量是在全局做用域(global做用域)上的,就要使用global先聲明一下,代碼以下:

count = 10
def outer():
    global count
    print(count) 
    count = 100
    print(count)
outer()
#10
#100
# global 能少用就少用,由於會對全局變量作出修改,影響全局其餘地方用這個全局變量


五、nonlocal關鍵字

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新增的關鍵字,有了這個關鍵字,就能完美的實現閉包了。閉包跟裝飾器有關係,在裝飾器裏介紹。

相關文章
相關標籤/搜索