翼度科技»论坛 编程开发 python 查看内容

Python 中 eval 与 exec 的相同点和不同点

6

主题

6

帖子

18

积分

新手上路

Rank: 1

积分
18
相同点

在 Python 中,eval 和 exec 都可以用来执行动态生成(dynamically generated)的代码。
两者在Python 3中的函数声明基本相同,如下所示:
  1. eval(expression[, globals[, locals]])
  2. exec(object[, globals[, locals]])
复制代码
其中,输入参数中,globals 必须是字典(dict)类型,表示全局空间的变量,若未提供,则通过 globals() 方法获取全局变量,若提供的字典类型对象不包含名为__builtins__的键,则会在表达式解析前,插入这个键,其值设为内置模块 builtins 的引用;而 locals 参数可以是任何可映射类型的对象,表示局部空间的变量,若未提供,则通过 locals() 方法获取局部变量。
不同点

下面从关键字类型、第一个输入参数、内调 compile 函数 这 3 个方面,讨论 eval 和 exec 的不同之处。
1. 类型不同

eval 在 Python 2 和 Python 3 中都是函数(function);
而 exec 在 Python 2 中是语句(statement),在 Python 3 中是函数。
2. 第一个输入参数不同

eval 是 evaluate 的英文简写,只能用来计算单独一个 Python 表达式(expression)的值,返回值是这个表达式的执行结果;在 Python 中,表达式(expression)定义为可以在变量赋值中,进行赋值的对象:
  1. # An expression in Python is whatever you can have as the value in a variable assignment:
  2. a_variable = (anything you can put within these parentheses is an expression)
复制代码
而 exec 是 execute 的英文简写,用来执行 Python 语句(statement),如循环语句、try...except...异常处理语句、class 定义、函数定义等,无返回值,即返回值始终为 None。
基本示例,如下所示:
  1. >>> a = 5
  2. >>> eval('37 + a')   # it is an expression
  3. 42
  4. >>> exec('37 + a')   # it is an expression statement; value is ignored (None is returned)
  5. >>> exec('a = 47')   # modify a global variable as a side effect
  6. >>> a
  7. 47
  8. >>> eval('a = 47')  # you cannot evaluate a statement
  9. Traceback (most recent call last):
  10.   File "<stdin>", line 1, in <module>
  11.   File "<string>", line 1
  12.     a = 47
  13.       ^
  14. SyntaxError: invalid syntax
复制代码
3. compile 函数的模式不同

eval 和 exec 在输入字符串类型时,内部都会首先调用 compile 函数编译为 bytecode,eval 函数对应的模式是 'eval',而 exec 对应的模式是 'exec'。compile 函数用来将输入参数 source 编译为 code 对象,具体声明如下:
  1. compile(source, filename, mode, flags=0, dont_inherit=False, optimize=-1)
复制代码
其中,参数 source 通常为包含 Python 源码的字符串;参数 filename 应该给出从中读取代码的文件,或一些可识别的值,常常使用  ;参数 mode 指定读取的代码的编译类型:如果包含多个语句,采用 'exec' 模式,如果只包含单一表达式,则采用 'eval' 模式;可选参数 flags 和 dont_inherit 用来控制 future 模块语句影响源码编译;参数 optimize 指定编译器的优化级别。更多内容参见 compile 官方文档。
采用 'exec' 模式的 compile 函数可以编译包含任意数量语句的源码为 bytecode,隐含返回值总是 None;而采用 'eval' 模式的 compile 函数只可以编译单一表达式为 bytecode,并返回这个表达式的值。如果在 'eval' 模式下,compile 函数的输入源码中包含语句或任何超出了单一表达式的要求,则会抛出 SyntaxError 异常。
一些具体示例,如下:
  1. >>> eval(compile('20200926', '<string>', 'exec'))  # code returns None
  2. >>> eval(compile('20200926', '<string>', 'eval'))  # code returns 20200926
  3. 20200926
  4. >>> exec(compile('20200926', '<string>', 'eval'))  # code returns 20200926,
  5. >>>                                          # but ignored by exec
  6. #学习中遇到问题没人解答?小编创建了一个Python学习交流群:531509025
  7. >>> compile('for i in range(3): print(i)', '<string>', 'eval')
  8. Traceback (most recent call last):
  9.   File "<stdin>", line 1, in <module>
  10.   File "<string>", line 1
  11.     for i in range(3): print(i)
  12.       ^
  13. SyntaxError: invalid syntax
复制代码
实际上,eval 只接收单一表达式(eval accepts only a single expression),只是在字符串直接传递给 eval 函数时有效。此时内部会使用 compile(source, , 'eval') 编译为 bytecode。如果一个包含 bytecode 的 code 对象传递给 exec 或 eval ,它们的表现是相同,除了 exec 总是会返回 None。所以采用 eval 函数执行带有语句的字符串也是可以的,但需要首先使用 compile 函数将源码转为 bytecode,再传给 eval 方法。
具体示例,如下:
  1. >>> eval(compile('if 1: print("Hello")', '<string>', 'exec'))
  2. Hello
  3. >>>
复制代码
来源:https://www.cnblogs.com/djdjdj123/p/18321528
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!

举报 回复 使用道具