在Python中,如果处理不好变量使用的关系,就很容易出现:

UnboundLocalError: local variable 'n' referenced before assignment

这种错误。

一般而言,这种错误都是这么出现的:

n = 1


def test():
    n += 1
    print(n)
if __name__ == '__main__':
    test()

原因很简单:在代码尚未执行时,当某个函数(作用域)内,对某个外部变量进行了处理,而使其值改变,则Python会认为操作的变量是局部变量;在代码运行时,对于这个Python自己声明的没有实际值的局部变量,直接对其进行操作就会出现错误。

说的很绕,用例子来说明吧。


这样是正常情况:

n = 1


def test():
    print(n)
if __name__ == '__main__':
    test()

因为test()函数中,没有对n的值进行改变,所以在test()中,n仍然是全局变量n


但这样就有问题了:

n = 1


def test():
    n += 1
    print(n)

if __name__ == '__main__':
    test()

错误如下:

Traceback (most recent call last):
  File "C:\Users\zhantong\Desktop\test.py", line 9, in <module>
    test()
  File "C:\Users\zhantong\Desktop\test.py", line 5, in test
    n += 1
UnboundLocalError: local variable 'n' referenced before assignment

这是因为Python发现了test()函数对n的值进行了改变,所以此时n是局部变量,但n没有值,自然就出错了。


有一个细节:

n = 1


def test():
    print(n)
    n += 1
    print(n)

if __name__ == '__main__':
    test()

这时出现的错误:

Traceback (most recent call last):
  File "C:\Users\zhantong\Desktop\test.py", line 10, in <module>
    test()
  File "C:\Users\zhantong\Desktop\test.py", line 5, in test
    print(n)
UnboundLocalError: local variable 'n' referenced before assignment

区别于上面的情况,这时是test()函数中,第一个print()出错,充分说明Python在test()实际运行前,即已声明n为局部变量。


解决的办法很容易,用global声明,防止Python自行处理:

n = 1


def test():
    global n
    print(n)
    n += 1
    print(n)

if __name__ == '__main__':
    test()

得到的结果:

1
2

这里还有一点需要注意,listdict类型并不受上述问题影响

直接用代码说明吧,list

l = [1, 2, 3]


def test():
    print(l)
    l[0] += 1
    print(l)

if __name__ == '__main__':
    test()

结果:

[1, 2, 3]
[2, 2, 3]

dict也是类似的:

d = {
    'a': 1,
    'b': 2,
}


def test():
    print(d)
    d['a'] += 1
    print(d)

if __name__ == '__main__':
    test()

结果:

{'a': 1, 'b': 2}
{'a': 2, 'b': 2}

综上,在操作一般变量时,需要考虑全局变量与局部变量的关系,采取的解决方法是传参或者global声明;在操作listdict等对象时,虽然不需要考虑这些问题,但一定要注意可能的不小心修改了其值。