>백엔드 개발 >파이썬 튜토리얼 >Python eval 함수는 수학적 표현식을 동적으로 평가합니다.

Python eval 함수는 수학적 표현식을 동적으로 평가합니다.

王林
王林앞으로
2023-04-11 23:43:182279검색

​이 글에서 윤듀오 선생님은 다음 두 가지 측면에서 여러분과 함께 공부할 것입니다.

  • Python의 eval() 작동 방식
  • eval()을 사용하여 임의의 문자열 기반 또는 컴파일된 코드 기반 입력을 동적으로 계산하는 방법

Python eval 함수는 수학적 표현식을 동적으로 평가합니다.

또한 나중에 트윗에서 Python을 함께 사용하는 방법을 배울 것입니다.​ eval ()​ 수학적 표현식을 대화형으로 평가하는 애플리케이션을 코딩합니다. 이 예에서는 eval()에 대해 배운 모든 내용을 실제 문제에 적용해 보겠습니다.

Python의 eval()

내장된 Python eval()[1]을 사용하여 문자열 기반 또는 컴파일된 코드 기반 입력의 표현식을 동적으로 평가할 수 있습니다. eval()​에 문자열을 전달하면 함수는 이를 구문 분석하고 바이트코드[2]로 컴파일한 후 Python 표현식으로 평가합니다. 그러나 컴파일된 코드 객체로 eval()​을 호출하면 함수는 계산 단계만 수행하므로 동일한 입력으로 eval()을 여러 번 호출하면 매우 편리합니다.

Python의 eval()은 다음과 같이 정의됩니다.

eval(expression[, globals[, locals]])

이 함수에는 계산해야 하는 표현식이 포함된 표현식이라는 첫 번째 매개변수가 필요합니다. eval()에는 두 개의 선택적 매개변수도 필요합니다.

  1. globals
  2. locals

다음 콘텐츠에서는 이러한 매개변수가 무엇인지, eval()이 이를 사용하여 Python 표현식을 즉시 평가하는 방법을 알아봅니다.

참고: exec()[3]을 사용하여 Python 코드를 동적으로 실행할 수도 있습니다. eval()​과 exec()​의 주요 차이점은 eval()​은 표현식만 실행하거나 평가할 수 있는 반면, exec()는 Python 코드의 모든 부분을 실행할 수 있다는 것입니다.

첫 번째 매개변수: 표현식

eval()​ 첫 번째 매개변수는 표현식​이라고 하며, 함수의 문자열 기반 또는 컴파일 기반 입력을 저장하는 데 사용되는 필수 매개변수입니다. eval()​이 호출되면 표현식의 내용이 Python 표현식으로 평가됩니다. 아래는 문자열 기반 입력을 사용한 예입니다.

>>> eval("2 ** 8")
256
>>> eval("1024 + 1024")
2048
>>> eval("sum([8, 16, 32])")
56
>>> x = 100
>>> eval("x * 2")
200

문자열을 인수로 eval()​을 호출하면 함수는 입력 문자열에 대한 계산 결과를 반환합니다. 기본적으로 eval()​은 위 예의 x와 같은 전역 변수 이름에 액세스할 수 있습니다.

문자열 기반 표현식을 평가하기 위해 Python의 eval()은 다음 단계를 실행합니다.

  1. 식을 구문 분석
  2. 바이트 코드로 컴파일
  3. 파이썬 식으로 평가
  4. 계산 결과를 반환

eval()의 첫 번째 매개 변수 식은 이 함수만 작동한다는 점을 강조하는 식입니다. 복합적인 진술이 아닙니다 [4]. Python 문서에서는 표현식을 다음과 같이 정의합니다.

expression

특정 값으로 계산할 수 있는 구문입니다. 즉, 표현식은 리터럴, 이름, 속성 액세스, 연산자, 함수 호출 등 모두 값을 반환하는 표현식 요소의 누적입니다. 다른 많은 언어와 달리 모든 언어 구성이 표현식은 아닙니다. while과 같이 표현식으로 사용할 수 없는 명령문도 있습니다. 또한 할당은 표현식이 아니라 명령문이기도 합니다.

반면 Python 문에는 다음과 같은 정의가 있습니다.

statement

statement는 제품군(코드의 "블록")의 일부입니다. 명령문은 표현식이거나 if, while 또는 for와 같은 키워드가 있는 여러 구조 중 하나입니다.

복합문을 eval()​에 전달하면 SyntaxError가 발생합니다. 다음 예에서는 eval()을 사용하여 if 문을 실행합니다.

>>> x = 100
>>> eval("if x: print(x)")
File "<string>", line 1
if x: print(x)
^
SyntaxError: invalid syntax

위에 보고된 오류는 eval()이 표현식만 허용하기 때문입니다. if, for, while, import, def 또는 class와 같은 다른 문은 오류를 발생시킵니다.

참고: for 루프는 복합문이지만 for 키워드는 이해에도 사용될 수 있으며 이 경우 표현식으로 간주됩니다. for 키워드를 사용하더라도 파생은 eval()​을 사용하여 평가할 수 있습니다.

eval()은 할당 작업도 허용하지 않습니다.

>>> eval("pi = 3.1416")
File "<string>", line 1
pi = 3.1416
 ^
SyntaxError: invalid syntax

할당 작업을 eval()에 매개변수로 전달하면 SyntaxError가 발생합니다. 할당 작업은 표현식이 아닌 명령문이며 명령문은 eval()과 함께 사용할 수 없습니다.

파서가 입력 표현식을 이해하지 못하는 경우에도 SyntaxError가 발생합니다. 다음 예에서는 Python 구문을 위반하는 표현식을 평가합니다.

>>> # Incomplete expression
>>> eval("5 + 7 *")
File "<string>", line 1
5 + 7 *
^
SyntaxError: unexpected EOF while parsing

所以,不能把一个违反 Python 语法的表达式传给 eval()​ 。在上面的例子中,我们尝试计算一个不完整的表达式 ("5 + 7 *") 时抛出一个 SyntaxError,因为分析器不理解表达式的语法。

我们也可以把已编译的代码对象传递给 eval()​ 。因此可以使用函数 compile()[7] ,一个内置函数,可以将输入的字符串编译成代码对象[8] 或 AST 对象[9],这样就可以用 eval() 来计算它。

如何使用compile()的细节超出了本文的范围,但这里可以快速了解一下它的前三个必要参数。

source保存我们要编译的源代码。这个参数可以接受普通字符串、字节字符串[10]和AST对象。

filename给出读取代码的文件。如果我们要使用一个基于字符串的输入,那么这个参数的值应该是""。

mode指定了我们想得到哪种编译后的代码。如果我们想用eval()​来处理编译后的代码,那么这个参数应该被设置为"eval"。

我们可以使用 compile()​ 向eval()提供代码对象,而不是普通的字符串。

>>> # 算术运算
>>> code = compile("5 + 4", "<string>", "eval")
>>> eval(code)
9
>>> code = compile("(5 + 7) * 2", "<string>", "eval")
>>> eval(code)
24
>>> import math
>>> # 一个球体的体积
>>> code = compile("4 / 3 * math.pi * math.pow(25, 3)", "<string>", "eval")
>>> eval(code)
65449.84694978735

如果我们使用 compile()​ 来编译要传递给eval()​的表达式,那么eval()会经过以下步骤。

  1. 计算编译后的代码
  2. 返回计算的结果

如果使用基于编译码的输入调用 eval()​ ,那么该函数会执行计算步骤并立即返回结果。当需要多次计算同一个表达式时,这可能很方便。在这种情况下,最好预先编译表达式,并在随后调用 eval() 时重复使用产生的字节码。

如果我们事先编译了输入表达式,那么连续调用eval()将运行得更快,因为我们不会重复解析和编译的步骤。如果我们正在计算复杂的表达式,不需要的重复会导致高的CPU时间和过度的内存消耗。

第二个参数:globals

eval()​ 的第二个参数 globals,可选的,字典类型,为 eval()​ 提供一个全局命名空间。通过 globals 告诉 eval() 在计算表达式时要使用哪些全局变量名。

全局变量名是所有那些在当前全局范围或命名空间中可用的变量名。可以从代码的任何地方访问它们。

在字典中传递给 globals 的所有名字在执行时都可以提供给 eval()​ 。请看下面的例子,它展示了如何使用一个自定义的字典来为 eval() 提供一个全局命名空间。

>>> x = 100# 一个全局变量
>>> eval("x + 100", {"x": x})
200
>>> y = 200# 另一个全局变量
>>> eval("x + y", {"x": x})
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<string>", line 1, in <module>
NameError: name 'y' is not defined

如果为 eval()​ 的 globals 参数提供一个自定义字典,那么 eval()​ 将只接受这些名字作为 globals。在这个自定义字典之外定义的任何全局变量名都不能从 eval()​ 内部访问。这就是为什么当你试图在上述代码中访问 y 时,Python 会引发一个 NameError。传递给 globals 的字典不包括 y。

可以通过在字典中列出名字来插入 globals,然后这些名字在求值过程中就会出现。例如,如果在 globals 中插入了 y​,那么在上面的例子中对 "x + y" 的求值将如期进行。

>>> eval("x + y", {"x": x, "y": y})
300

因为把 y​ 添加到了自定义 globals 字典中,所以成功计算 "x + y" 的值,得到的预期返回值 300。

我们也可以提供不存在于当前全局范围的变量名。此时需要为每个名字提供一个具体的值。eval()在运行时将把这些变量名解释为全局变量名。

>>> eval("x + y + z", {"x": x, "y": y, "z": 300})
600
>>> z
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'z' is not defined

尽管z​没有在当前的全局范围内定义,但是这个变量在全局中的值是300,此时eval()​可以访问z,就像它是一个全局变量一样。

globals 背后的机制是相当灵活的,可以向 globals 传递任何可见的变量(全局、局部、或者非局部)。还可以传递自定义的键值对,比如上面例子中的 "z": 300​,那么eval() 将把它们全部作为全局变量处理。

关于 globals 中的注意事项,如果我们提供给它的自定义字典不包含键值 "__builtins__​",那么在表达式被解析之前,对内置字典的引用将自动插入 "__builtins__​" 下面。这可以确保 eval() 在计算表达式时可以完全访问所有的 Python 内置变量名。

下面的例子表明,即使给 globals 提供了一个空的字典,对 eval() 的调用仍然可以访问 Python 的内置变量名。

>>> eval("sum([2, 2, 2])", {})
6
>>> eval("min([1, 2, 3])", {})
1
>>> eval("pow(10, 2)", {})
100

在上面的代码中,我们向 globals 提供了一个空的字典 ({}​)。由于这个字典不包含一个叫做 "__builtins__​" 的键,Python 会自动插入一个指向 builtins 中名字的引用。这样,eval() 在解析表达式时就可以完全访问所有 Python 的内置名字。

如果调用 eval()​ 而没有将自定义字典传递给 globals ,那么参数将默认为在调用 eval()​的环境中 globals() 返回的字典:

>>> x = 100#一个全局变量
>>> y = 200# 另一个全局变量
>>> eval("x + y")# 访问两个全局变量
300

当调用 eval()​ 而不提供 globals 参数时,该函数使用 globals()​ 返回的字典作为其全局命名空间来计算表达式。所以,在上面的例子中,我们可以自由地访问 x​ 和 y,因为它们是包含在我们当前全局范围内的全局变量。

第三个参数:locals

Python 的 eval()​ 第三个参数 locals ,可选参数,字典类型。此时这个字典包含了 eval() 在计算表达式时作为局部变量名使用的变量。

局部变量名是那些我们在一个给定的函数内定义的名称(变量、函数、类等等)。局部名称只在封闭的函数内可见。我们在编写函数时定义这些变量名。

因为 eval()​ 已经写好了,所以不能在它的代码或局部范围内添加局部变量名。然而可以向 locals​ 传递一个字典,eval()会把这些名字当作本地名字。

>>> eval("x + 100", {}, {"x": 100})
200
>>> eval("x + y", {}, {"x": 100})
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<string>", line 1, in <module>
NameError: name 'y' is not defined

第一个调用 eval()​ 的第二个字典保存了变量 x​。这个变量被 eval()​ 解释为一个局部变量。换句话说,它被看作是在 eval() 中定义的一个变量。

我们可以在表达式中使用 x​,并且 eval()​ 可以访问它。相反,如果使用y​,那么会得到一个 NameError,因为y没有定义在 globals 命名空间或 locals 命名空间。

和 globals 一样,可以向 locals 传递任何可见的变量(全局、局部或非局部)。也可以传递自定义的键值对,比如 "x"​。eval()将把它们全部作为局部变量处理。

注意,要给 locals 提供一个字典,首先需要给 globals 提供一个字典。不能在 eval() 中使用关键字参数。

>>> eval("x + 100", locals={"x": 100})
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: eval() takes no keyword arguments

如果在调用 eval()​ 时使用关键字参数,那么抛出一个 TypeError。这是因为 eval() 不接受关键字参数,所以在提供 locals 字典之前,需要先提供一个 globals 字典。

如果没有给 locals 传递一个字典,那么它就默认为传递给 globals 的字典。这里有一个例子,给 globals 传递了一个空的字典,而 locals 没有传递任何值。

>>> x = 100
>>> eval("x + 100", {})
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<string>", line 1, in <module>
NameError: name 'x' is not defined

鉴于没有给 locals 提供一个自定义的字典,这个参数默认为传递给 globals 的字典。此时eval()​ 无法访问 x,因为 globals 持有一个空的字典。

globals 和 locals 之间的主要实际区别是,如果"__builtins__​"键不存在,Python 会自动插入 globals 中。无论我们是否为 globals 提供了一个自定义的字典,这都会发生。此外,如果我们给 locals 提供了一个自定义的字典,那么在执行 eval() 的过程中,这个字典将保持不变。

用 eval() 计算表达式

我们可以使用Python的eval()来计算任何一种Python表达式,但不包括Python语句,如基于关键字的复合语句或赋值语句。

当我们需要动态地计算表达式,而使用其它 Python 技术或工具会大大增加我们的开发时间和精力时,eval() 可以很方便。

在这一节中,我们将学习如何使用 Python 的 eval() 来计算布尔、数学和通用的 Python 表达式。

布尔表达式

布尔表达式 是Python表达式,当解释器对其进行计算时返回一个真值(True​ 或者 False​)。它们通常用在if语句中,以检查某些条件是否为真或假。由于布尔表达式不是复合语句,我们可以使用eval()来计算它们。

>>> x = 100
>>> y = 100
>>> eval("x != y")
False
>>> eval("x < 200 and y > 100")
False
>>> eval("x is y")
True
>>> eval("x in {50, 100, 150, 200}")
True

我们可以用 eval() 来处理使用以下任何Python运算符的布尔表达式。

  • 值比较运算符: , =, ==, !=
  • 逻辑(布尔)运算符:and​,or​,not
  • 成员测试运算符:in​,not in
  • 身份运算符:is​,is not

在所有情况下,该函数都会返回正在计算的表达式的真值。

我们思考,为什么我应该使用eval()而不是直接使用布尔表达式呢?假设需要实现一个条件语句,但我们想临时改变条件。

>>> def func(a, b, condition):
... if eval(condition):
... return a + b
... return a - b
...
>>> func(2, 4, "a > b")
-2
>>> func(2, 4, "a < b")
6
>>> func(2, 2, "a is b")
4

在func()​中,使用eval()​来计算所提供的条件,并根据计算的结果返回a+b​或a-b​。在上面的例子中,只使用了几个不同的条件,但还可以使用任何数量的其他条件,只要坚持使用我们在func()​中定义的名称a​和b。

现在想象一下,如果不使用Python的eval(),我们将如何实现这样的东西。那会花更少的代码和时间吗?不可能!

数学表达式

Python 的 eval()​ 的一个常见用例是对基于字符串的输入进行 math​ 表达式的计算。例如,创建一个 Python 计算器,那么可以使用 eval() 来计算用户的输入并返回计算结果。

下面的例子演示了如何使用eval()​与数学一起进行math运算。

>>> # Arithmetic operations
>>> eval("5 + 7")
12
>>> eval("5 * 7")
35
>>> eval("5 ** 7")
78125
>>> eval("(5 + 7) / 2")
6.0
>>> import math
>>> # 一个圆的面积
>>> eval("math.pi * pow(25, 2)")
1963.4954084936207
>>> # 球体的体积
>>> eval("4 / 3 * math.pi * math.pow(25, 3)")
65449.84694978735
>>> # 直角三角形的斜边
>>> eval("math.sqrt(math.pow(10, 2) + math.pow(15, 2))")
18.027756377319946

当我们使用eval()​来计算数学表达式时,我们可以传入任何种类或复杂程度的表达式,eval()会解析它们,计算它们,如果一切正常,就会给我们预期结果。

通用表达式

前面我们已经学会了如何在布尔和 math​ 表达式中使用 eval()​ 。然而,我们可以在更复杂的 Python 表达式中使用 eval() ,这些表达式包括函数调用、对象创建、属性访问、列表推导式等等。

例如,可以调用一个内置函数或用标准或第三方模块导入的函数。

>>> # 运行echo命令
>>> import subprocess
>>> eval("subprocess.getoutput('echo Hello, World')")
'Hello, World'
>>> # 启动Firefox(如果有的话)
>>> eval("subprocess.getoutput('firefox')")
''

在这个例子中,我们使用 Python 的 eval()​ 来执行一些系统命令。我们可以用这个功能做大量有用的事情。然而,eval()也会有一些严重的安全风险,比如允许一个恶意的用户在我们的机器中运行系统命令或任何任意的代码。

위 내용은 Python eval 함수는 수학적 표현식을 동적으로 평가합니다.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
이 기사는 51cto.com에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제