Python中變量的作用域

轉載:http://www.2cto.com/kf/201405/300359.html
1、作用域介紹
python中的作用域分4種情況: L:local,局部作用域,即函數中定義的變量;
E:enclosing,嵌套的父級函數的局部作用域,即包含此函數的上級函數的局部作用域,但不是全局的;
G:globa,全局變量,就是模塊級別定義的變量; B:built-in,系統固定模塊裏面的變量,比如int, bytearray等。 搜索變量的優先級順序依次是:作用域局部>外層作用域>當前模塊中的全局>python內置作用域,也就是LEGB。

x = int(2.9)  # int built-in
g_count = 0  # global
def outer():
    o_count = 1  # enclosing
    def inner():
        i_count = 2  # local

當然,local和enclosing是相對的,enclosing變量相對上層來說也是local。
2、作用域產生
在Python中,只有模塊(module),類(class)以及函數(def、lambda)纔會引入新的作用域,其它的代碼塊(如if、try、for等)是不會引入新的作用域的,如下代碼:

if True:
    x = 1;
print(x)

這個是沒有問題的,if並沒有引入一個新的作用域,x仍處在當前作用域中,後面代碼可以使用。

def test():
    x2 = 2
print(x2)
# NameError: name 'x2' is not defined

def、class、lambda是可以引入新作用域的。
3、變量的修改
一個不在局部作用域裏的變量默認是隻讀的,如果試圖爲其綁定一個新的值,python認爲是在當前的局部作用域裏創建一個新的變量,也就是說在當前局部作用域中,如果直接使用外部作用域的變量,那麼這個變量是隻讀的,不能修改,如:

count = 10
def outer():
    print(count)  
    count = 100
    print(count)
outer()
#UnboundLocalError: local variable 'count' referenced before assignment

這裏第一個print中,使用到了外部作用域的count,這樣後面count就指外部作用域中的count了,再修改就會報錯。 如果沒使用過這個變量,而直接賦值,會認爲是新定義的變量,此時會覆蓋外部作用域中變量,如:

count = 10
def outer():
    count = 100  
    print(count)
outer()
#100

內部作用域中直接聲明瞭count=100,後面使用count都是內部作用域的了。
4、global關鍵字
當內部作用域想修改外部作用域的變量時,就要用到global和nonlocal關鍵字了,當修改的變量是在全局作用域(global作用域)上的,就要使用global先聲明一下,代碼如下:

count = 10
def outer():
    global count
    print(count)  
    count = 100
    print(count)
outer()
#10
#100

5、nonlocal關鍵字
global關鍵字聲明的變量必須在全局作用域上,不能嵌套作用域上,當要修改嵌套作用域(enclosing作用域,外層非全局作用域)中的變量怎麼辦呢,這時就需要nonlocal關鍵字了

def outer():
    count = 10
    def inner():
        nonlocal count
        count = 20
        print(count)
    inner()
    print(count)
outer()

#20
#20

6、小結
(1)變量查找順序:LEGB,作用域局部>外層作用域>當前模塊中的全局>python內置作用域; (2)只有模塊、類、及函數才能引入新作用域; (3)對於一個變量,內部作用域先聲明就會覆蓋外部變量,不聲明直接使用,就會使用外部作用域的變量; (4)內部作用域要修改外部作用域變量的值時,全局變量要使用global關鍵字,嵌套作用域變量要使用nonlocal關鍵字。nonlocal是python3新增的關鍵字,有了這個關鍵字,就能完美的實現閉包了。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章