標籤:
Python的範圍解析是基於叫做LEGB(Local(本地),Enclosing(封閉),Global(全域),Built-in(內建))的規則進行操作的。這看起來很直觀,對吧?事實上,在Python中這有一些細微的地方很容易出錯。看這個例子:
1
2
3
4
5
6
7
8
9
10
>>> x = 10
>>> def foo():
... x += 1
... print x
...
>>> foo()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 2, in foo
UnboundLocalError: local variable ‘x‘ referenced before assignment
這是怎麼回事?
這是因為,在一個範圍裡面給一個變數賦值的時候,Python自動認為這個變數是這個範圍的本地變數,並屏蔽範圍外的同名的變數。
很多時候可能在一個函數裡添加一個賦值的語句會讓你從前本來工作的代碼得到一個UnboundLocalError。(感興趣的話可以讀一讀這篇文章。)
在使用列表(lists)的時候,這種情況尤為突出。看下面這個例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
>>> lst = [1, 2, 3]
>>> def foo1():
... lst.append(5) # 這沒有問題...
...
>>> foo1()
>>> lst
[1, 2, 3, 5]
>>> lst = [1, 2, 3]
>>> def foo2():
... lst += [5] # ... 這就有問題了!
...
>>> foo2()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 2, in foo
UnboundLocalError: local variable ‘lst‘ referenced before assignment
嗯?為什麼foo2有問題,而foo1沒有問題?
答案和上一個例子一樣,但是更加不易察覺。foo1並沒有給lst賦值,但是foo2嘗試給lst賦值。注意lst+=[5]只是lst=lst+[5]的簡寫,由此可以看到我們嘗試給lst賦值(因此Python假設範圍為本地)。但是,這個要賦給lst的值是基於lst本身的(這裡的範圍仍然是本地),而lst卻沒有被定義,這就出錯了。
http://blog.jobbole.com/68256/
Python範圍的規則小析