当一个变量/函数在一个内部定义时,exec
它似乎转向locals()
了globals()
我如何改变这种行为?仅当您将全局和本地字典传递给exec
.
例子:
exec("""
a = 2
def foo():
print a
foo()""") in {},{}
当你尝试这个时:
NameError: global name 'a' is not defined
乍一看,这对我来说也很奇怪。但是有了更多的输出,我找到了原因:
>>> g, l = {}, {}
>>> print id(g), id(l)
12311984 12310688
>>>
>>> exec '''
... a = 2
... print 'a' in globals(), 'a' in locals(), id(globals()), id(locals())
... def f():
... print 'a' in globals(), 'a' in locals(), id(globals()), id(locals())
... f()
... ''' in g, l
False True 12311984 12310688
False False 12311984 12311264
如http://effbot.org/pyfaq/what-are-the-rules-for-local-and-global-variables-in-python.htm所述:
在 Python 中,仅在函数内部引用的变量是隐式全局的。如果一个变量在函数体内的任何地方都被赋予了一个新值,那么它就被认为是一个局部变量。如果一个变量在函数内部被赋予了一个新值,那么该变量是隐式本地的,您需要将其显式声明为全局。
因此,一种解决方案是对全局变量和本地变量使用相同的 dict:
>>> l = {}
>>> exec '''
... a = 2
... def f():
... print a
... f()
... ''' in l, l
2
如果a
不是全球性的,那就让它成为全球性的......
exec("""
global a
a = 2
def foo():
print a
foo()""") in {}, {}