В 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
является свободной переменной, поэтому она компилирует ее в глобальную ссылку:
>>> 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
для вас, я полагаю ... они не должны быть хорошими.