可能重复:
Why is not the keyword global not required in this case?
我想知道为什么我可以在没有global
关键字的情况下更改全局词典?为什么它对其他类型是强制性的?这背后有什么逻辑吗?
E.g。代码:
#!/usr/bin/env python3
stringvar = "mod"
dictvar = {'key1': 1,
'key2': 2}
def foo():
dictvar['key1'] += 1
def bar():
stringvar = "bar"
print(stringvar)
print(dictvar)
foo()
print(dictvar)
print(stringvar)
bar()
print(stringvar)
给出以下结果:
me@pc:~/$ ./globalDict.py
{'key2': 2, 'key1': 1}
{'key2': 2, 'key1': 2} # Dictionary value has been changed
mod
bar
mod
我期待的地方:
me@pc:~/$ ./globalDict.py
{'key2': 2, 'key1': 1}
{'key2': 2, 'key1': 1} # I didn't use global, so dictionary remains the same
mod
bar
mod
答案 0 :(得分:38)
原因是该行
stringvar = "bar"
是不明确的,它可能是指一个全局变量,或它可能会创建一个名为stringvar
的新局部变量。在这种情况下,Python默认假设它是局部变量,除非已使用global
关键字。
然而,行
dictvar['key1'] += 1
完全毫不含糊。它只能引用全局变量dictvar
,因为语句必须已存在dictvar
而不会抛出错误。
这不是字典特有的 - 列表也是如此:
listvar = ["hello", "world"]
def listfoo():
listvar[0] = "goodbye"
或其他类型的对象:
class MyClass:
foo = 1
myclassvar = MyClass()
def myclassfoo():
myclassvar.foo = 2
时都是如此
答案 1 :(得分:6)
您可以在不使用global
关键字的情况下修改任何可变对象。
这在Python中是可行的,因为当您想要将新对象重新分配给已在全局范围中使用的变量名称或定义新的全局变量时,会使用global
。
但是在可变对象的情况下你不会重新分配任何东西,你只需要就地修改它们,因此Python只是从全局范围加载它们并修改它们。
正如docs所说:
如果没有全局变量,就无法分配给全局变量。
In [101]: dic = {}
In [102]: lis = []
In [103]: def func():
dic['a'] = 'foo'
lis.append('foo') # but fails for lis += ['something']
.....:
In [104]: func()
In [105]: dic, lis
Out[105]: ({'a': 'foo'}, ['foo'])
dis.dis
:
In [121]: dis.dis(func)
2 0 LOAD_CONST 1 ('foo')
3 LOAD_GLOBAL 0 (dic) # the global object dic is loaded
6 LOAD_CONST 2 ('a')
9 STORE_SUBSCR # modify the same object
3 10 LOAD_GLOBAL 1 (lis) # the global object lis is loaded
13 LOAD_ATTR 2 (append)
16 LOAD_CONST 1 ('foo')
19 CALL_FUNCTION 1
22 POP_TOP
23 LOAD_CONST 0 (None)
26 RETURN_VALUE