python 2.7 / exec / что не так? - PullRequest
5 голосов
/ 06 августа 2010

У меня есть этот код, который отлично работает в Python 2.5, но не в 2.7:

import sys
import traceback
try:
    from io import StringIO
except:
    from StringIO import StringIO

def CaptureExec(stmt):
    oldio = (sys.stdin, sys.stdout, sys.stderr)
    sio = StringIO()
    sys.stdout = sys.stderr = sio
    try:
        exec(stmt, globals(), globals())
        out = sio.getvalue()
    except Exception, e:
        out = str(e) + "\n" + traceback.format_exc()
    sys.stdin, sys.stdout, sys.stderr = oldio
    return out

print "%s" % CaptureExec("""
import random
print "hello world"
""")

И я получаю:

string argument expected, got 'str'
Traceback (most recent call last):
  File "D:\3.py", line 13, in CaptureExec
    exec(stmt, globals(), globals())
  File "", line 3, in 
TypeError: string argument expected, got 'str'

Ответы [ 2 ]

14 голосов
/ 06 августа 2010

io.StringIO сбивает с толку в Python 2.7, потому что он перенесен из мира 3.x байтов / строк.Этот код выдает ту же ошибку, что и ваша:

from io import StringIO
sio = StringIO()
sio.write("Hello\n")

вызывает:

Traceback (most recent call last):
  File "so2.py", line 3, in <module>
    sio.write("Hello\n")
TypeError: string argument expected, got 'str'

Если вы используете только Python 2.x, пропустите модуль io и вставьтес помощью StringIO.Если вы действительно хотите использовать io, измените ваш импорт на:

from io import BytesIO as StringIO
2 голосов
/ 06 августа 2010

Это плохие новости

io.StringIO хочет работать с юникодом.Вы можете подумать, что можете это исправить, поместив u перед строкой, которую вы хотите напечатать следующим образом:

print "%s" % CaptureExec("""
import random
print u"hello world"
""")

однако print действительно не работает, поскольку вызывает 2 записи в StringIO,Первый - u"hello world", что нормально, но затем следует "\n"

, поэтому вместо этого вам нужно написать что-то вроде этого

print "%s" % CaptureExec("""
import random
sys.stdout.write(u"hello world\n")
""")
...