Python:全局变量与局部变量的问题
在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
这里还有一点需要注意,list
和dict
类型并不受上述问题影响
直接用代码说明吧,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
声明;在操作list
、dict
等对象时,虽然不需要考虑这些问题,但一定要注意可能的不小心修改了其值。