標籤:代碼 too global 效果 用法 現象 注意 var 全域變數
Python 中,一個變數的範圍總是由在代碼中被賦值的地方所決定的。
- 函數定義了本地範圍,而模組定義的是全域範圍。
如果想要在函數內定義全域範圍,需要加上global修飾符。
- 變數名解析:LEGB原則
當在函數中使用未認證的變數名時,Python搜尋4個範圍[本地範圍(L)(函數內部聲明但沒有使用global的變數),之後是上一層結構中def或者lambda的本地範圍(E),之後是全域範圍(G)(函數中使用global聲明的變數或在模組層聲明的變數),最後是內建範圍(B)(即python的內建類和函數等)]並且在第一處能夠找到這個變數名的地方停下來。如果變數名在整個的搜尋過程中
都沒有找到,Python就會報錯。
補:上面的變數規則只適用於簡單對象,當出現引用對象的屬性時,則有另一套搜尋規則:屬性引用搜尋一個或多個對象,而不是範圍,並且有可能涉及到所謂的"繼承"
- 以上基於http://blog.csdn.net/carolzhang8406/article/details/6855525
- 下面探討global修飾符的用法:
- 首先是pythond的一個奇異現象,在模組層面定義的變數(無需global修飾),如果在函數中沒有再定義同名變數,可以在函數中當做全域變數使用:
hehe=6def f(): print(hehe)f()print(hehe)
上述代碼可以正常運行且輸出為6和6
- 但如果在函數中有再賦值/定義(因為python是弱類型語言,指派陳述式和其定義變數的語句一樣),則會產生引用了未定義變數的錯誤:
-
hehe=6def f(): print(hehe) hehe=2f()print(hehe)
拋出的錯誤資訊為:
- UnboundLocalError: local variable ‘hehe‘ referenced before assignment
而如果在函數中的定義在引用前使用,那麼會正常運行但函數中的變數和模組中定義的全域變數不為同一個
-
hehe=6def f(): hehe=2 print(hehe)f()print(hehe)
上述輸出是2和6,也即f函數中print使用的是局部變數hehe,而最後一個print語句使用的全域hehe。
- 那麼我們會有疑問,如果我可能在函數使用某一變數後又對其進行修改(也即再賦值),怎麼讓函數裡面使用的變數是模組層定義的那個全域變數而不是函數內部的局部變數呢?這時候global修飾符就派上用場了。
-
hehe=6def f(): global hehe print(hehe) hehe=3f()print(hehe)
在用global修飾符聲明hehe是全域變數的hehe後(注意,global語句不允許同時進行賦值如global hehe=3是不允許的),上述輸出是6和3,得到了我們想要的效果。
Python中的範圍及global用法