为何在exec内关闭闭包?
在Python 2.6中,
>>> exec "print (lambda: a)()" in dict(a=2), {}
2
>>> exec "print (lambda: a)()" in globals(), {'a': 2}
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<string>", line 1, in <module>
File "<string>", line 1, in <lambda>
NameError: global name 'a' is not defined
>>> exec "print (lambda: a).__closure__" in globals(), {'a': 2}
None
我希望它打印2
两次,然后打印一个元组cell
。与3.1中的情况相同。这是怎么回事?
-
当您将字符串传递给
exec
或时eval
,它将在考虑全局变量或局部变量之前将该字符串编译为代码对象。所以当你说:eval('lambda: a', ...)
它的意思是:
eval(compile('lambda: a', '<stdin>', 'eval'), ...)
没有办法
compile
知道这a
是一个freevar,因此将其编译为全局引用:>>> c= compile('lambda: a', '<stdin>', 'eval') >>> c.co_consts[0] <code object <lambda> at 0x7f36577330a8, file "<stdin>", line 1> >>> dis.dis(c.co_consts[0]) 1 0 LOAD_GLOBAL 0 (a) 3 RETURN_VALUE
因此,要使其发挥作用,您必须
a
输入全局变量而不是本地变量。是的,这有点狡猾。但是那是对的
exec
,eval
我想…对于他们来说,它们应该不是很好。