深入理解Python中命名空间的查找规则LEGB


Posted in Python onAugust 06, 2015

名字空间

Python 的名字空间是 Python 一个非常核心的内容。
其他语言中如 C 中,变量名是内存地址的别名,而在 Python 中,名字是一个字符串对象,它与他指向的对象构成一个{name:object}关联。
Python 由很多名字空间,而 LEGB 则是名字空间的一种查找规则。
作用域

Python 中name-object的关联存储在不同的作用域中,各个不同的作用域是相互独立的。而我们就在不同的作用域中搜索name-object。

举个栗子,来说明作用域是相互独立的。

In [11]: i = "G"

In [12]: def test():
      i = "L"
      print i, "in locals"
  ....:

In [13]: test()
    L in locals

In [14]: print i, "in globals"
    G in globals

 

在上面的栗子中,我们定义了两次 i,在 test 函数中是 i-L,在外面是 i-G。为什么在 test 函数中,我们 i 指向的是对象 L,而在外面,i 指向的则是 G?这就是 LEGB 的作用。
简述

简而言之,LEGB 代表名字查找顺序: locals -> enclosing function -> globals -> __builtins__

  •     locals 是函数内的名字空间,包括局部变量和形参
  •     enclosing 外部嵌套函数的名字空间(闭包中常见)
  •     globals 全局变量,函数定义所在模块的名字空间
  •     builtins 内置模块的名字空间

所以,在 Python 中检索一个变量的时候,优先回到 locals 里面来检索,检索不到的情况下会检索 enclosing ,enclosing 没有则到 globals 全局变量里面检索,最后是到 builtins 里面来检索。

当然,因为 builtins 的特殊性,我们可以直接在 builtins 里面添加变量,这样就可以在任意模块中访问变量,不过这种方法太过于变态,不推荐这么做。
locals,globals

函数的形参跟内部变量都存储在 locals 中。

In [1]: def f(x):
  ...:   a = x
  ...:   print a
  ...:   print locals()
  ...:


In [2]: f("hello")
hello
{'a': 'hello', 'x': 'hello'}

不过在函数内部调用global 声明的时候,可以将变量存储在 globals 中

In [6]: def f(x):
  ...:   global a
  ...:   a = x
  ...:   print a
  ...:   print locals()
  ...:

In [7]: f("hello")
hello
{'x': 'hello'}

In [8]: print a
hello

In [9]: print x
---------------------------------------------------------------------------
NameError                 Traceback (most recent call last)
<ipython-input-9-2d264e11d975> in <module>()
----> 1 print x

NameError: name 'x' is not defined

如上面栗子中那样,在函数中声明 a 为全局变量,则函数 f 的 locals只有参数 x,而没有变量,而在外部可以使用变量 a,而使用 x 的时候则是NameError
Enclosed

Enclosing 是外部嵌套函数的名字空间。我们经常在闭包中用到。在 Python3中提供了一个 nonlocal关键字来修改外部嵌套函数的名字空间,但是要使用 Python3才有,我等使用 Python2的只能眼馋一下。

In [11]: def outer():
  ....:   a_var = 'enclosed value'
  ....:   print a_var
  ....:   def inner():
  ....:     a_var = 'local value'
  ....:     print(a_var)
  ....:   inner()
  ....:   print a_var
  ....:

In [12]: outer()
enclosed value
local value
enclosed value

下面的栗子简单示范一下 nonlocal 的用法,实在 Python3下面才可以正常运行的:

In [1]: a_var = 'global value'

In [2]: def outer():
  ...:   a_var = "local value"
  ...:   print("outer befor", a_var)
  ...:   def inner():
  ...:     nonlocal a_var
  ...:     a_var = "inner value"
  ...:     print("in inner():", a_var)
  ...:   inner()
  ...:   print("outer inner:", a_var)
  ...:

In [3]: outer()
outer befor local value
in inner(): inner value
outer inner: inner value

In [4]: print(a_var)
global value

builtins

builtins 则是内置模块,轻易不要修改

In [19]: b
---------------------------------------------------------------------------
NameError                 Traceback (most recent call last)
<ipython-input-19-3b5d5c371295> in <module>()
----> 1 b

NameError: name 'b' is not defined

In [20]: __builtins__.b = "builtins"

In [21]: b
Out[21]: 'builtins'

上面栗子中在第一次调用b的时候报错NameError,之后我们修改 builtins 的名字空间,将名字b与值"builtins"进行关联,就可以正常调用了。这种非常规用法不建议使用。

Python 相关文章推荐
java直接调用python脚本的例子
Feb 16 Python
Pandas之drop_duplicates:去除重复项方法
Apr 18 Python
python3+PyQt5+Qt Designer实现堆叠窗口部件
Apr 20 Python
python中yaml配置文件模块的使用详解
Apr 27 Python
python设计tcp数据包协议类的例子
Jul 23 Python
wxpython绘制圆角窗体
Nov 18 Python
Pytorch之contiguous的用法
Dec 31 Python
python数据库操作mysql:pymysql、sqlalchemy常见用法详解
Mar 30 Python
python mock测试的示例
Oct 19 Python
详解基于python的图像Gabor变换及特征提取
Oct 26 Python
tensorflow中的数据类型dtype用法说明
May 26 Python
Python 游戏大作炫酷机甲闯关游戏爆肝数千行代码实现案例进阶
Oct 16 Python
举例详解Python中yield生成器的用法
Aug 05 #Python
Python中return语句用法实例分析
Aug 04 #Python
python函数形参用法实例分析
Aug 04 #Python
Python简明入门教程
Aug 04 #Python
将Python代码打包为jar软件的简单方法
Aug 04 #Python
python函数局部变量用法实例分析
Aug 04 #Python
python删除列表内容
Aug 04 #Python
You might like
PHP获取当前页面完整URL的实现代码
2013/06/10 PHP
php 调用ffmpeg获取视频信息的简单实现
2017/04/03 PHP
PHP实现webshell扫描文件木马的方法
2017/07/31 PHP
深入解析PHP底层机制及相关原理
2020/12/11 PHP
如何实现JS函数的重载
2006/09/22 Javascript
Javascript事件热键兼容ie|firefox
2010/12/30 Javascript
让textarea自动调整大小的js代码
2011/04/12 Javascript
JS实现金额转换(将输入的阿拉伯数字)转换成中文的实现代码
2013/09/30 Javascript
js获取电脑分辨率的思路及操作
2013/11/22 Javascript
javascript内置对象操作详解
2015/02/04 Javascript
JS实现简易图片轮播效果的方法
2015/03/25 Javascript
简单对比分析JavaScript中的apply,call与this的使用
2015/12/04 Javascript
第一次接触Bootstrap框架
2016/10/24 Javascript
移动端利用H5实现压缩图片上传功能
2017/03/29 Javascript
JavaScript屏蔽Backspace键的实现代码
2017/11/02 Javascript
Vue根据条件添加click事件的方式
2019/11/09 Javascript
深入理解 TypeScript Reflect Metadata
2019/12/12 Javascript
原生JS实现pc端轮播图效果
2020/12/21 Javascript
Python创建文件和追加文件内容实例
2014/10/21 Python
python通过线程实现定时器timer的方法
2015/03/16 Python
详解python 字符串和日期之间转换 StringAndDate
2017/05/04 Python
浅谈tensorflow1.0 池化层(pooling)和全连接层(dense)
2018/04/27 Python
Python实现将数据写入netCDF4中的方法示例
2018/08/30 Python
python3+requests接口自动化session操作方法
2018/10/13 Python
谈一谈基于python的面向对象编程基础
2019/05/21 Python
Python aiohttp百万并发极限测试实例分析
2019/10/26 Python
TensorFlow2.X结合OpenCV 实现手势识别功能
2020/04/08 Python
Python库skimage绘制二值图像代码实例
2020/04/10 Python
使用 prometheus python 库编写自定义指标的方法(完整代码)
2020/06/29 Python
Python 如何调试程序崩溃错误
2020/08/03 Python
优秀大学生求职自荐信范文
2014/04/19 职场文书
英语三分钟演讲稿
2014/08/19 职场文书
Golang之sync.Pool使用详解
2021/05/06 Golang
面试必问:圣杯布局和双飞翼布局的区别
2021/05/13 HTML / CSS
pytorch 权重weight 与 梯度grad 可视化操作
2021/06/05 Python
九大龙王魂骨,山龙王留下躯干骨,榜首死的最憋屈(被捏碎)
2022/03/18 国漫