Andr? Roberge wrote:

I have the following two files:

#--testexec.py--
def exec_code(co):
    try:
        exec co
    except:
        print "error"

#-- test.py--
import thread
import testexec
import time

code = "def a():\n print 'a'\n\n" +\
       "def c():\n a()\n\nc()"

code2 = "def a():\n print 'a'\n\n" +\
       "def c():\n global a\n a()\n\nc()"

print " exec code - no global"
exec code
print " exec from thread - no global"
thread.start_new(testexec.exec_code, (code,))
time.sleep(1)
print "\n exec code2 - with global"
exec code2
print " exec from thread - with global"
thread.start_new(testexec.exec_code, (code2,))
#-----------------------

Here's the output when I execute test.py:

 exec code - no global
a
 exec from thread - no global
error

 exec code2 - with global
a
 exec from thread - with global
a
#---------
Without the global statement, I get an error when trying to execute
the code.
I don't understand why I need to use the global statement within the
definition of c() in order for it to know what a() is.  If I define
exec_code() within test.py and use it there, I do not get any error,
with or without the use of a global statement.

Andre

I have taken the liberty of restructuring your program slightly, by using a "from" to explicitly import the function you need, and by using extended string literals (""" ... """) to make the code easier to read. I have also included print statements to show the contents of the local and global namespaces after the definition of function a(). After these transformations it looks like this:


import thread
from  testexec import exec_code
import time

code = """\
def a():
  print 'a'

def c():
  a()

c()
"""

code2 = """\
def a():
  print 'a'

def c():
 global a
 a()

c()
"""

print " exec code - no global"
exec code
print " exec from thread - no global"
thread.start_new(exec_code, (code,))
time.sleep(1)
print "\n exec code2 - with global"
exec code2
print " exec from thread - with global"
thread.start_new(exec_code, (code2,))
time.sleep(1)

(OK, I added a final sleep so I saw the output from the second thread).

The reason you are seeing this behavior lies in the behavior of the exec statement. The full syntax for that statement is

exec_stmt  ::=  "exec" expression ["in" expression ["," expression]]

The second and third expressions are mappings that will be used as namespaces. Since you don't provide either, the interpreter uses the current scope (whose contents can be determined using the locals() function) for both namespaces, so it doesn't matter whether the function is added to the local or the global namespace.

When using threads, however (and, by the way, the usual advice is to use the threading module, which has a more convenient API), the namespaces are clearly different. The a() function is being added to the local namespace for the exec.

So ultimately it's to do with namespaces, as many of the more perplexing problems in Python are. I hope this helps establish exactly *why* you see what you do.

regards
 Steve
--
http://www.holdenweb.com
http://pydish.holdenweb.com
Holden Web LLC +1 800 494 3119
--
http://mail.python.org/mailman/listinfo/python-list

Reply via email to