经典错误
1
以下这段代码是在python中使用闭包时一段经典的错误代码
1
2
3
4
5
6
|
def foo(): a = 1 def bar(): a = a + 1 return a return bar |
这段程序的本意是要通过在每次调用闭包函数时都对变量a进行递增的操作。但在实际使用时
1
2
3
4
5
6
|
>>> c = foo() >>> print c() Traceback (most recent call last): File "<stdin>" , line 1 , in <module> File "<stdin>" , line 4 , in bar UnboundLocalError: local variable 'a' referenced before assignment |
这是因为在执行代码 c = foo()时,python会导入全部的闭包函数体bar()来分析其的局部变量,python规则指定所有在赋值语句左面的变量都是局部变量,则在闭包bar()中,变量a在赋值符号"="的左面,被python认为是bar()中的局部变量。再接下来执行print c()时,程序运行至a = a + 1时,因为先前已经把a归为bar()中的局部变量,所以python会在bar()中去找在赋值语句右面的a的值,结果找不到,就会报错。解决的方法很简单,在python3以后,在a = a + 1 之前,使用语句nonlocal a就可以了,该语句显式的指定a不是闭包的局部变量。