Python作用域

python作用域

LEGB原則

python中作用域有四種:

L (Local) 局部作用域
E (Enclosing) 閉包函數外的函數中
G (Global) 全局作用域
B (Built-in) 內建作用域

python按照LEGB原則搜索變量,即優先級L>E>G>B。

#dir 爲python內建函數
dir = 1 # Global
def outer():
    dir = 2  # Enclosing
    def inner():
        dir = 3 # Local
        return dir
    return inner

print outer()() # 輸出3

作用域(Scope)和命名空間(NameSpace)

def/lambda會創建新的作用域,生成器表達式都有引入新的作用域,class的定義沒有作用域,只是創建一個隔離的命名空間。在Python中,scope是由namespace按特定的層級結構組合起來的。scope一定是namespace,但namespace不一定是scope。命名空間跟作用域的區別是,它不能在裏面再嵌套其他作用域。下面看兩個例子。

例1

a = 1
def test():
    a += 1
    a = 2

test() #異常

UnboundLocalError: local variable ‘a’ referenced before assignment。這是因爲解釋器看到a+=1時,按照LEGB優先在Local中找到了a的聲明,執行時先a+=1在a=2聲明之前,所有拋出異常。

例2

class A(object):
    x = 2
    gen = (x*i for i in xrange(5))

if __name__ == "__main__":
    a = A()
    print list(a.gen)#異常

上面的代碼會拋出異常:NameError: global name ‘x’ is not defined。這是因爲gen = (x for _ in xrange(5)是生成器,會產生新的作用域。而classA 中並不產生作用域。按照LEGB原則,不能找到x的定義,所以拋出異常。解決這個問題有幾種方案。
1,將x定義爲全局變量,這樣可以解決異常,但是可能違背了類的邏輯。
2,將生成器表達式改爲列表表達式。

gen = [x*i for i in xrange(5)]

在python2中,列表表達式不產生新的作用域,所以不會拋出異常。但是在python3中仍有異常。
3,用A.x的方式訪問類屬性。

gen = (A.x*i for i in xrange(5))

4,引入lambda函數,將class命名空間的x作爲變量傳入到匿名函數中。

gen = (lambda x: (x*i for i in xrange(5)))(x)

這個問題可以理解爲class不能產生作用域導致的,在函數中就沒有這個問題。

def test():
    x = 2
    gen = (i * x for i in xrange(5))
    return gen

gen = test()
print list(gen)#輸出[0, 2, 4, 6, 8]
發佈了46 篇原創文章 · 獲贊 10 · 訪問量 11萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章