compile() 函数将一个字符串编译为字节代码。
以下是 compile() 方法的语法:
compile(source, filename, mode[, flags[, dont_inherit]])
参数
source – 字符串或者AST抽像语法树(Abstract Syntax Trees)对象。
filename – 代码文件名称,如果不是从文件读取代码则传递一些可辨认的值。参数filename用于在执行代码报错的运行时错误消息中显示该参数对应的信息,当source是执行代码从文件中读取的代码字符串时,则可以存放文件名,如果不是从文件里读取源码来编译,那么这里可以放一些用来标识这些代码的字符串,其值理论上是任何字符串,没有特殊要求,一般都放‘’,用于表示前面的source是个字符串,如果source放AST,则可以标识为‘’;
mode – 指定编译代码的种类。可以指定为 exec, eval, single。exec’表示编译的是一段代码或模块, 'single’表示编译的是一个单独的语句, 'eval’表示编译的是一个表达式而不是一个语句
flags – 变量作用域,局部命名空间,如果被提供,可以是任何映射对象。。
flags和dont_inherit是用来控制编译源码时的标志
compile 函数返回结果
1、如果编译通过,结果可以生成字节码(类型code)或者AST(抽像语法树),字节码可以使用函数exec()或eval来执行,而AST可以使用eval()来继续编译(关于AST的内容本节都不介绍,ATS 对象:Abstract Syntax Tree,抽象语法树,是源代码语法结构的一种抽象表示。关于抽象语法树大家可以参考:https://zhuanlan.zhihu.com/p/26988179;
2、exec 语句:exec 执行储存在字符串或文件中的Python语句,相比于 eval,exec可以执行更复杂的 Python 代码。需要说明的是在 Python2 中exec不是函数,而是一个内置语句;
3、如果编译的源码不合法,此函数会触发 SyntaxError 异常;如果源码包含 空字节(空字符串),则3.5版本以前会触发 ValueError 异常,3.5版本后则不会触发可以编译通过并执行。注意:
先执行一个简单的代码段, 代码段写到一个字符串
a = """
x = "hello"
print(x)
"""f = compile(a, '', 'exec')
exec(f) # 执行代码段
# hello
再举个for 循环代码段
b = """
x = ["hello", "world"]
for i in x:print(i)
"""f = compile(b, '', 'exec')
exec(f) # 执行代码段
# hello
# world
代码段也可以是一个函数
b = """
def func(x):print("xxx" + x)return x+"hello"func("world")
"""f = compile(b, '', 'exec')
exec(f) # 执行代码段
# xxxworld
eval 也可以把字符串当代码去执行
# 表达式
c = "3+5"
print(eval(c))
# 8
eval 只能执行一个表达式,不能执行代码段
# 代码段无法执行
c = """
x = "hello"
print(x)
"""
print(eval(c))
上面的代码段无法执行会报错
eval和exec有这两个区别:
在’exec’模式下的编译将任意数量的语句编译成一个隐式总是返回None的字节码,而在’eval’模式下,它将单个表达式编译为返回该表达式的值的字节码。
>>> eval(compile('42', '', 'exec')) # code return None
>>> eval(compile('42', '', 'eval')) # code returns 42
42
>>> exec(compile('42', '', 'eval')) # code returns 42,
>>> # but ignored by exec