In 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
I expected it to print 2
twice, and then print a tuple with a single cell
. It is the same situation in 3.1. What's going on?
When you pass a string to
exec
oreval
, it compiles that string to a code object before considering globals or locals. So when you say:it means:
There's no way for
compile
to know thata
is a freevar, so it compiles it to a global reference:Therefore to make it work you have to put
a
in the globals and not the locals.Yeah, it's a bit dodgy. But then that's
exec
andeval
for you I suppose... they're not supposed to be nice.