为什么在exec中关闭了闭包?

Dev*_*rre 17 python closures 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
Run Code Online (Sandbox Code Playgroud)

我希望它打印2两次,然后用单个打印元组cell.这与3.1中的情况相同.这是怎么回事?

bob*_*nce 25

将字符串传递给exec或时eval,它会在考虑全局变量或局部变量之前将该字符串编译为代码对象.所以当你说:

eval('lambda: a', ...)
Run Code Online (Sandbox Code Playgroud)

它的意思是:

eval(compile('lambda: a', '<stdin>', 'eval'), ...)
Run Code Online (Sandbox Code Playgroud)

没有办法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        
Run Code Online (Sandbox Code Playgroud)

因此,为了使它工作,你必须放入a全局变量而不是本地变量.

是的,它有点狡猾.但那就是那个execeval你在一起我想......他们不应该是好人.