Python | LEGB规则

LEGB的名字取自:

  • Locals :当前命名空间,如函数或模块
  • Enclosing :外部嵌套函数的命名空间(闭包常见)
  • Globals:全局变量的命名空间
  • Builtins : 内建模块的命名空间

各作用域首字母

LEGB规则是:Python 使用 LEGB 的顺序来查找一个符号对应的对象
顺序是:
locals -> enclosing function -> globals -> builtins
查找顺序从左到右,不会反过来,比如一个功能在globals,当它在当前作用域找不到定义时,只能从builtins查找,而不会从enclosing中去查找

下面以一道题来演示

def proc1():
    j, k = 3, 4
    print('j == %d and k == %d' % (j, k))
    k = 5

def proc2():
    j = 6
    proc1()
    print('j == %d and k == %d' % (j, k))
    
k = 7

proc1()  #1

#print('j == %d and k == %d' % (j, k))  #2

j = 8

proc2()  # 3

print('j == %d and k == %d' % (j, k))  # 4

从代码中那个我们可以看出,一共有 #1和#3两个函数调用,#2和#4两个print语句(下面会说#2为何要注释掉)
在看以下分析前,你可以自己先想下答案是什么

  • #1

Python函数从上往下运行语句,proc1中print了 jk ,print前定义了 j, k = 3, 4,因此 #1 的结果输出是

j == 3 and k == 4
  • #2

现在我们来说为何#2要被注释掉了,在#2 print语句之前,只定义了k的值,而没有定义 j 的值。如果不注释掉会报错

NameError: name 'j' is not defined

有人估计会问,#1 中不是定义了 j 的值吗?
因为proc1函数中的 j、k 值只作用在 Locals 中,而 #2 中的 j、k 明显是 Globals 中参数,Globals 是不会从 Locals 去搜寻值的,只能从 builtins 中去寻找,所以“ j is not defined ”。

  • #3

这里值得注意的是, proc2 中先是 定义了 j 值,然后调用了 proc1(),再 print 语句,所以这里会有2个输出
前面说,Python函数从上往下运行语句,那么 proc2 中 proc1 中的 j 值会变成 6 吗?
这里虽然 j 是 proc2 的Locals参数,但 proc1 中有自己的Locals,为何要从 proc2 中去 “借”呢?(如果proc1 从 proc2 中取值,那是闭包的用法了)

所以 proc1() 的结果是:

j == 3 and k == 4

随后的 print 结果:

j == 6 and k == 7

j 是 自己定义的,k 是 Globals 的,如果这里看不懂 k 为何是7,请从头再认真看一遍

  • #4

这个没什么好说的:

j == 8 and k == 7

Summary | 小结

  • 寻找顺序:locals -> enclosing function -> globals -> builtins
  • 官大一级压死人,事情安排找“下级”
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容