Python exec中的范围

时间:2012-07-04 09:22:19

标签: python scope exec globals locals

当在exec中定义变量/函数时,它似乎转到locals()而不是globals()我如何更改此行为?只有在将全局和本地词典传递给exec时才会发生这种情况。

示例:

exec("""
a = 2

def foo():
    print a

foo()""") in {},{}

当你尝试这个时:

NameError: global name 'a' is not defined

2 个答案:

答案 0 :(得分:2)

乍一看,这对我来说很奇怪。但是有了更多的输出我找到了原因:

>>> 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

答案 1 :(得分:-1)

如果a不是全局的,请将其设为全局...

exec("""
global a
a = 2

def foo():
    print a

foo()""") in {}, {}