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版本后则不会触发可以编译通过并执行。注意:
在 'single' 或 'eval' 模式编译多行代码字符串(这些串必须是一个完整语句或表达式而不是多个语句或表达式)时,输入必须以至少一个换行符结尾;
2)如果编译足够大或者足够复杂的字符串成 AST 对象时,Python 解释器会因为 Python AST 编译器的栈深度限制而崩溃
先执行一个简单的代码段, 代码段写到一个字符串
a = """
x = "hello"
print(x)
f = compile(a, '<string>', 'exec')
exec(f) # 执行代码段
# hello
再举个for 循环代码段
b = """
x = ["hello", "world"]
for i in x:
print(i)
f = compile(b, '<string>', 'exec')
exec(f) # 执行代码段
# hello
# world
代码段也可以是一个函数
b = """
def func(x):
print("xxx" + x)
return x+"hello"
func("world")
f = compile(b, '<string>', 'exec')
exec(f) # 执行代码段
# xxxworld
eval() 函数的使用
eval 也可以把字符串当代码去执行
# 表达式
c = "3+5"
print(eval(c))
eval 只能执行一个表达式,不能执行代码段
# 代码段无法执行
c = """
x = "hello"
print(x)
print(eval(c))
上面的代码段无法执行会报错
eval和exec有这两个区别:
eval只接受一个表达式,exec可以使用具有Python语句的代码块:循环,try:except :,类和函数/方法定义等。
eval返回给定表达式的值,而exec忽略其代码的返回值,并始终返回None(在Python 2中,它是一个语句,不能用作表达式,因此它不会返回任何内容)。
'exec' 模式 与 'eval'模式
在'exec'模式下的编译将任意数量的语句编译成一个隐式总是返回None的字节码,而在'eval'模式下,它将单个表达式编译为返回该表达式的值的字节码。
>>> eval(compile('42', '<string>', 'exec')) # code return None
>>> eval(compile('42', '<string>', 'eval')) # code returns 42
>>> exec(compile('42', '<string>', 'eval')) # code returns 42,
>>> # but ignored by exec