Python里只有2种作用域:全局作用域和局部作用域。全局作用域是指当前代码所在模块的作用域,局部作用域是指当前函数或方法所在的作用域。其实准确来说,Python 3.x引入了nonlocal关键字,可以用于标识外部作用域的变量。 局部作用域里的代码可以读外部作用域(包括全局作用域)里的变量,但不能更改它。一旦进行更改,就会将其当成是局部变量。而如果在更改前又进行了读取操作,则会抛出异常。
Python中(2.X和3.X版本)如果在一个函数的局部作用域中修改外部作用域的变量,就会报UnboundLocalError
错误:
In [6]: help()
Welcome to Python 2.7! This is the online help utility.
....
In [7]: def foo():
...: c = 1
...: def inner():
...: c += 1
...: return c
...: return inner
...:
In [8]: print foo()()
---------------------------------------------------------------------------
UnboundLocalError Traceback (most recent call last)
<ipython-input-8-1ec86e0af5ba> in <module>()
----> 1 print foo()()
<ipython-input-7-8107f285e6fe> in inner()
2 c = 1
3 def inner():
----> 4 c += 1
5 return c
6 return inner
UnboundLocalError: local variable 'c' referenced before assignment
如果想要对外部作用域里面的变量进行修改,最简单的办法就是将其放入全局作用域,用global关键字引入该变量。
In [9]: def foo():
...: global c
...: c = 1
...: def inner():
...: global c
...: c += 1
...: return c
...: return inner
In [10]: print foo()()
2
在Python 2.x版本中中,闭包只能读外部函数的变量,而不能改写它。
In [11]: def a():
...: x =0
...: def b():
...: y = x +1
...: print locals() # 返回当前位置的所有局部变量;locals()函数是只读的.
...: print x,y
...: return b
...:
In [12]: print a()()
{'y': 1, 'x': 0} # 可以看到返回值中,x变量的值并没有修改,也不能修改。
0 1
None
如果要对x进行赋值操作,在Python 2.x中解决这个问题,目前只能使用全局变量:global或者像上面的这个例子一样曲线救国。
为了解决这个问题,Python 3.x引入了nonlocal关键字(详见The nonlocal statement)。只要在闭包内用nonlocal声明变量,就可以让解释器在外层函数中查找变量名了.
In [13]: def a():
...: x = 0
...: def b():
...: nonlocal x
...: x += 1
...: print(x)
...: return b
In [14]: print(a()())
1
None