名字空间
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__
所以,在 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命名空间详解,包括了Python命名空间详解的使用技巧和注意事项,需要的朋友参考一下 通俗的来说,Python中所谓的命名空间可以理解为一个容器。在这个容器中可以装许多标识符。不同容器中的同名的标识符是不会相互冲突的。理解python的命名空间需要掌握三条规则: 第一,赋值(包括显式赋值和隐式赋值)产生标识符,赋值的地点决定标识符所处的命名空间。 第二,函数定义(包括de
1. 命名空间的定义 命名空间 (Namespace) 是从名称到值的映射,大部分的命名空间都是通过 Python 字典来实现的,它的键就是变量名,它的值是变量的值。 1.1 例子 一个包含 3 个变量的命名空间,如下图所示: 图: 定义了 3 个变量的命名空间 第一个变量 名为 a,值为 1 第二个变量 名为 b,值为 2 第三个变量 名为 c,值为 3 1.2 在同一个命名空间中,不允许重名
有时候也想要一个任务的行为是基于已经定义好的取值范围或者特定规则, 下面的例子就提供了一种很直观漂亮的方式: 例子 15.25. 任务规则 build.gradle tasks.addRule("Pattern: ping<ID>") { String taskName -> if (taskName.startsWith("ping")) { task(taskName)
如果我删除所有命名空间和前缀,我的XSLT转换将完美运行。 我曾尝试过几次引入名称空间,但最终总是出现某种错误或没有输出,因此我在这里寻求一些帮助和理解。 我的预期输出,缩减为命名空间和前缀,是: 再次缩减的XML输入是: 为了实现正确的命名空间转换,我需要向XSLT头中添加什么?
问题内容: 命名空间Python程序包(no )和常规Python程序包(具有)之间有什么区别,尤其是对于常规程序包为空时? 我很好奇,因为最近我一直忘了自己制作的包装,而且我从没发现任何问题。实际上,它们的行为似乎与常规软件包相同。 编辑:命名空间包仅受Python 3.3支持(请参阅PEP 420 ),因此自然地,此问题仅适用于Python 3。 问题答案: 阅读Aaron和PEP420的链接
控制器 只允许使用 a-z、A-Z、0-9 和 _ ,并以大写字母开头,例如:Index.php 模型 只允许使用 a-z、A-Z、0-9 和 _ ,并以大写字母开头,例如:Info.php 视图 只允许使用 a-z、0-9 和 _ ,并以字母开头,例如:index.php 布局 只允许使用 a-z、A-Z、0-9 和 _ ,并以字母开头,例如:header.php API接口 只允许使用 a-z