Why can I not change global variables from inside a function, using exec()? It works fine when the assignment statement is outside of exec(). Here is an example of my proble
How about this:
>>> myvar = 'test'
>>> def myfunc():
... exec('globals()["myvar"] = "changed!"')
... print(myvar)
...
>>> myfunc()
changed!
>>> print(myvar)
changed!
It worked for me in Python 2.6.
EDIT: Actually Alex Martelli's explanation is much better than mine :)
Per the docs, the exec
statement takes two optional expressions, defaulting to globals()
and locals()
, and always performs changes (if any) in the locals()
one.
So, just be more explicit/specific/precise...:
>>> def myfunc():
... exec('myvar="boooh!"', globals())
...
>>> myfunc()
>>> myvar
'boooh!'
...and you'll be able to clobber global variables to your heart's contents.
To add to Alex's answer: although when you omit the locals/globals arguments they default to the locals and globals of the caller, this only a convenience hack; it does not mean they are inheriting the full execution context of the caller. In particular:
a. nested scope cells are not available to the execed code. So this fails:
def f():
foo= 1
def g():
exec('print foo')
g()
f()
b. global
declarations do not carry over into the execed code. So by default as in your example, written variables are put in the locals dictionary. However, you could make it work by saying
exec('global myvar\nmyvar = "changed!"')
You don't really want to be doing this if you can help it. global
already isn't nice and exec
is pretty much a code smell in itself! You wouldn't want to combine them unless there was really no alternative.