Ответ 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
для вас я предполагаю... они не должны быть хорошими.