前言

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
文章目录