乐趣区

关于python:深入理解Python-5个硬核函数

看到一篇很好的对 Python 中的 eval()、exec()及其相干函数总结的一篇博文,第一工夫分享给你。

eval, exec, compile, locals, globals 这些函数对于深刻了解 Python 很有必要,心愿认真看看。

刚好前些天有人提到 eval()与 exec()这两个函数,所以就翻了下 Python 的文档。这里就来简略说一下这两个函数以及与它们相干的几个函数,如 globals()、locals()和 compile():

1. eval 函数

函数的作用:

计算指定表达式的值。也就是说它要执行的 Python 代码只能是单个运算表达式(留神 eval 不反对任意模式的赋值操作),而不能是简单的代码逻辑,这一点和 lambda 表达式比拟类似。

函数定义:

eval(expression, globals=None, locals=None)

参数阐明:

  • expression:必选参数,能够是字符串,也能够是一个任意的 code 对象实例(能够通过 compile 函数创立)。如果它是一个字符串,它会被当作一个(应用 globals 和 locals 参数作为全局和本地命名空间的)Python 表达式进行剖析和解释。
  • globals:可选参数,示意全局命名空间(寄存全局变量),如果被提供,则必须是一个字典对象。
  • locals:可选参数,示意以后部分命名空间(寄存局部变量),如果被提供,能够是任何映射对象。如果该参数被疏忽,那么它将会取与 globals 雷同的值。
  • 如果 globals 与 locals 都被疏忽,那么它们将取 eval()函数被调用环境下的全局命名空间和部分命名空间。

返回值:

  • 如果 expression 是一个 code 对象,且创立该 code 对象时,compile 函数的 mode 参数是 ’exec’,那么 eval()函数的返回值是 None;
  • 否则,如果 expression 是一个输入语句,如 print(),则 eval()返回后果为 None;
  • 否则,expression 表达式的后果就是 eval()函数的返回值;

实例:

x = 10

def func():
    y = 20
    a = eval('x + y')
    print('a:', a)
    b = eval('x + y', {'x': 1, 'y': 2})
    print('b:', b)
    c = eval('x + y', {'x': 1, 'y': 2}, {'y': 3, 'z': 4})
    print('c:', c)
    d = eval('print(x, y)')
    print('d:', d)

func()

输入后果:

a:  30
b:  3
c:  4
10 20
d:  None

对输入后果的解释:

  • 对于变量 a,eval 函数的 globals 和 locals 参数都被忽略了,因而变量 x 和变量 y 都获得的是 eval 函数被调用环境下的作用域中的变量值,即:x = 10, y = 20,a = x + y = 30
  • 对于变量 b,eval 函数只提供了 globals 参数而疏忽了 locals 参数,因而 locals 会取 globals 参数的值,即:x = 1, y = 2,b = x + y = 3
  • 对于变量 c,eval 函数的 globals 参数和 locals 都被提供了,那么 eval 函数会先从全副作用域 globals 中找到变量 x, 从部分作用域 locals 中找到变量 y,即:x = 1, y = 3, c = x + y = 4
  • 对于变量 d,因为 print()函数不是一个计算表达式,没有计算结果,因而返回值为 None

2. exec 函数

函数的作用:

动静执行 Python 代码。也就是说 exec 能够执行简单的 Python 代码,而不像 eval 函数那么样只能计算一个表达式的值。

函数定义:

exec(object[, globals[, locals]])

参数阐明:

  • object:必选参数,示意须要被指定的 Python 代码。它必须是字符串或 code 对象。如果 object 是一个字符串,该字符串会先被解析为一组 Python 语句,而后在执行(除非产生语法错误)。如果 object 是一个 code 对象,那么它只是被简略的执行。
  • globals:可选参数,同 eval 函数
  • locals:可选参数,同 eval 函数

返回值:

exec 函数的返回值永远为 None.

须要阐明的是在 Python 2 中 exec 不是函数,而是一个内置语句 (statement),然而 Python 2 中有一个 execfile() 函数。能够了解为 Python 3 把 exec 这个 statement 和 execfile()函数的功可能整合到一个新的 exec()函数中去了:

eval()函数与 exec()函数的区别:

  • eval()函数只能计算单个表达式的值,而 exec()函数能够动静运行代码段。
  • eval()函数能够有返回值,而 exec()函数返回值永远为 None。

实例 1:

咱们把实例 1 中的 eval 函数换成 exec 函数试试:

x = 10

def func():
    y = 20
    a = exec('x + y')
    print('a:', a)
    b = exec('x + y', {'x': 1, 'y': 2})
    print('b:', b)
    c = exec('x + y', {'x': 1, 'y': 2}, {'y': 3, 'z': 4})
    print('c:', c)
    d = exec('print(x, y)')
    print('d:', d)

func()

输入后果:

a:  None
b:  None
c:  None
10 20
d:  None

因为咱们说过了,exec 函数的返回值永远为 None。

实例 2:

x = 10
expr = """
z = 30
sum = x + y + z
print(sum)
"""
def func():
    y = 20
    exec(expr)
    exec(expr, {'x': 1, 'y': 2})
    exec(expr, {'x': 1, 'y': 2}, {'y': 3, 'z': 4})
    
func()

输入后果:

60
33
34

对输入后果的解释:

前两个输入跟下面解释的 eval 函数执行过程一样,不做过多解释。对于最初一个数字 34,咱们能够看出是:x = 1, y = 3 是没有疑难的。对于 z 为什么还是 30 而不是 4,这其实也很简略,咱们只须要无理一下代码执行过程就能够了,其执行过程相当于:

x = 1
y = 2

def func():
    y = 3
    z = 4
    
    z = 30
    sum = x + y + z
    print(sum)

func()

3. globals()与 locals()函数

函数定义及性能阐明:

先来看下这两个函数的定义和文档形容

globals() 

形容:  Return a dictionary representing the current global symbol table. This is always the dictionary of the current module (inside a function or method, this is the module where it is defined, not the module from which it is called).

翻译:  返回一个示意以后全局标识符表的字典。这永远是以后模块的字典(在一个函数或办法外部,这是指定义该函数或办法的模块,而不是调用该函数或办法的模块)

locals()

形容:  Update and return a dictionary representing the current local symbol table. Free variables are returned by locals() when it is called in function blocks, but not in class blocks.

Note The contents of this dictionary should not be modified; changes may not affect the values of local and free variables used by the interpreter.

翻译:  更新并返回一个示意以后部分标识符表的字典。自在变量在函数外部被调用时,会被 locals()函数返回;自在变量在类累不被调用时,不会被 locals()函数返回。

留神:  locals()返回的字典的内容不应该被扭转;如果肯定要扭转,不应该影响被解释器应用的局部变量和自在变量。

总结:

  • globals()函数以字典的模式返回的定义该函数的模块内的全局作用域下的所有标识符(变量、常量等)
  • locals()函数以字典的模式返回以后函数内的局域作用域下的所有标识符
  • 如果间接在模块中调用 globals()和 locals()函数,它们的返回值是雷同的

实例 1:

name = 'Tom'
age = 18

def func(x, y):
    sum = x + y
    _G = globals()
    _L = locals()
    print(id(_G), type(_G),  _G)
    print(id(_L), type(_L), _L)

func(10, 20)

输入后果:

2131520814344 <class 'dict'> {'__builtins__': <module 'builtins' (built-in)>, 'func': <function func at 0x000001F048C5E048>, '__doc__': None, '__file__': 'C:/Users/wader/PycharmProjects/LearnPython/day04/func5.py', '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x000001F048BF4C50>, '__spec__': None, 'age': 18, '__name__': '__main__', 'name': 'Tom', '__package__': None, '__cached__': None}
2131524302408 <class 'dict'> {'y': 20, 'x': 10, '_G': {'__builtins__': <module 'builtins' (built-in)>, 'func': <function func at 0x000001F048C5E048>, '__doc__': None, '__file__': 'C:/Users/wader/PycharmProjects/LearnPython/day04/func5.py', '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x000001F048BF4C50>, '__spec__': None, 'age': 18, '__name__': '__main__', 'name': 'Tom', '__package__': None, '__cached__': None}, 'sum': 30}

实例 2:

name = 'Tom'
age = 18

G = globals()
L = locals()
print(id(G), type(G), G)
print(id(L), type(L), L)

输入后果:

2494347312392 <class 'dict'> {'__file__': 'C:/Users/wader/PycharmProjects/LearnPython/day04/func5.py', '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x00000244C2E44C50>, 'name': 'Tom', '__spec__': None, '__builtins__': <module 'builtins' (built-in)>, '__cached__': None, 'L': {...}, '__package__': None, '__name__': '__main__', 'G': {...}, '__doc__': None, 'age': 18}
2494347312392 <class 'dict'> {'__file__': 'C:/Users/wader/PycharmProjects/LearnPython/day04/func5.py', '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x00000244C2E44C50>, 'name': 'Tom', '__spec__': None, '__builtins__': <module 'builtins' (built-in)>, '__cached__': None, 'L': {...}, '__package__': None, '__name__': '__main__', 'G': {...}, '__doc__': None, 'age': 18}

下面打印出的 G 和 L 的内存地址是一样的,阐明在模块级别 locals()的返回值和 globals()的返回值是雷同的。

4. compile 函数

函数的作用:

将 source 编译为 code 对象或 AST 对象。code 对象可能通过 exec()函数来执行或者通过 eval()函数进行计算求值。

函数定义:

compile(source, filename, mode[, flags[, dont_inherit]])

参数阐明:

  • source:字符串或 AST(Abstract Syntax Trees)对象,示意须要进行编译的 Python 代码
  • filename:指定须要编译的代码文件名称,如果不是从文件读取代码则传递一些可辨认的值(通常是用 ”)
  • mode:用于标识必须当做那类代码来编译;如果 source 是由一个代码语句序列组成,则指定 mode=’exec’;如果 source 是由单个表达式组成,则指定 mode=’eval’;如果 source 是由一个独自的交互式语句组成,则指定 mode=’single’。
  • 另外两个可选参数暂不做介绍

实例:

s = """
for x in range(10):
    print(x, end='')
print()
"""code_exec = compile(s,'<string>','exec')
code_eval = compile('10 + 20', '<string>', 'eval')
code_single = compile('name = input("Input Your Name: ")', '<string>', 'single')

a = exec(code_exec)
b = eval(code_eval)

c = exec(code_single)
d = eval(code_single)

print('a:', a)
print('b:', b)
print('c:', c)
print('name:', name)
print('d:', d)
print('name;', name)

输入后果:

0123456789
Input Your Name: Tom
Input Your Name: Jerry
a:  None
b:  30
c:  None
name:  Jerry
d:  None
name;  Jerry

5. 这几个函数的关系

comiple()函数、globals()函数、locals()函数的返回后果能够当作 eval()函数与 exec()函数的参数应用。

另外,咱们能够通过判断 globals()函数的返回值中是否蕴含某个 key 来判断,某个全局变量是否曾经存在(被定义)。

最近刚好整顿了 Python 书籍材料电子版等材料,都打包好了,须要的小伙伴可间接返回公众号【Python 编程学习圈】支付,关注回复关键词【学习材料】即可收费获取,蕴含有入门到进阶、爬虫、数据分析等方向的材料教程,不要错过啦!

退出移动版