本文说下本身对 Python 命名空间和做用域的理解。html
注意:内容基于 Python 3.6python
A namespace is a mapping from names to objects.后端
命名空间,直译是名称到对象(好比数字、字符串等)的映射,(个人理解是)这些名称构成一个命名空间。通常有三种命名空间bash
看个例子就清楚了app
# A 是全局名称, object 是内置名称
class A(object):
a = -1 # a 是局部名称(类中)
# print_abs_a 是局部名称(类中), self 是局部名称(函数中)
def print_abs_a(self):
# temp_str 是局部名称(函数中), abs 是内置名称
temp_str = '%s 的绝对值是 %s' % (self.a, abs(self.a))
print(temp_str) # print 是内置名称
t = A() # t 是全局名称
复制代码
命名空间的生命周期各不相同函数
A scope is a textual region of a Python program where a namespace is directly accessible. “Directly accessible” here means that an unqualified reference to a name attempts to find the name in the namespace.ui
做用域,是 Python 代码中的一段文本区域,在这个区域里能「直接」访问一个命名空间中的名称。所谓「直接」,就是只要给出名称(如 some_name
)就能找到命名空间中的对应的名称,而不须要使用相似 modulename.subname 或是 object.attribute 等这样的方式。spa
有四种做用域:code
其中,须要强调的是 local scope 和 nonlocal scope 是一个相对的概念。若是一个模块中,函数 A 直接包含了函数 B , B 又直接包含了函数 C 。若是以 C 中的名称做为参考,那么 C 中的做用域为 local scope,则 B 中的做用域就为 nonlocal scope。若是以 B 中的名称做为参考,那么 B 中的做用域是 local scope, 则 A 中的做用域为 nonlocal。若是以 A 中的名称做为参考,那么 A 中的做用域是 local scope,不过要注意,模块中的做用域始终为 global scope,这时并无 nonlocal scope。htm
对于赋值操做,默认都是操做当前做用域中包含的名称。假设如今在 local scope,若是要对 nonlocal scope 包含的名称进行赋值,则要用 nonlocal 关键字。若是要对 global scope 中包含的名称赋值要用 global 关键字。须要注意的是,若是在某个做用域内没有对应的名称,则在对应的做用域中会新增。 下面的例子能够帮你理解赋值操做
def scope_test():
def do_local():
spam = "local spam"
def do_nonlocal():
nonlocal spam
spam = "nonlocal spam"
def do_global():
global spam
spam = "global spam"
spam = "test spam"
do_local()
print("After local assignment:", spam)
do_nonlocal()
print("After nonlocal assignment:", spam)
do_global()
print("After global assignment:", spam)
scope_test()
print("In global scope:", spam)
复制代码
输出为
After local assignment: test spam
After nonlocal assignment: nonlocal spam
After global assignment: nonlocal spam
In global scope: global spam
复制代码
对应读值操做,都是由内到外进行搜索。即 local scope -> nonlocal scope -> global scope -> builtin scope,若是都找不到对应的名称,则报错。
>>> x = 1
>>> def t():
... def tt():
... print(x)
... tt()
...
>>> t()
1
>>> del x
>>> t()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 4, in t
File "<stdin>", line 3, in tt
NameError: name 'x' is not defined
复制代码
若是要读取指定做用域的名称,则可使用对应的 nonlocal 或 global 关键字,若是对应做用域找不到该名称,则直接报错。
>>> x = 1
>>> def t():
... x = 2
... def tt():
... global x
... print(x)
... tt()
...
>>> t()
1
>>> del x
>>> t()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 6, in t
File "<stdin>", line 5, in tt
NameError: name 'x' is not defined
复制代码
固然,若是当前做用域已有同名的名称,就不能使用这 nonlocal 或 global 了,不然会报错。
>>> x = 1
>>> def t():
... x = 2
... global x
...
File "<stdin>", line 3
SyntaxError: name 'x' is assigned to before global declaration
复制代码
同时发现个有趣的地方
>>> def t():
... def tt():
... nonlocal x
... print(x)
... tt()
...
File "<stdin>", line 3
SyntaxError: no binding for nonlocal 'x' found
>>> def t():
... def tt():
... global x
... print(x)
... tt()
...
>>> t()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 5, in t
File "<stdin>", line 4, in tt
NameError: name 'x' is not defined
复制代码
第一个报的是语法错误,而第二个是运行时报的错误。这说明了一个问题:局部名称的查找是编译时就肯定的,而全局名称和内置名称的查找都是在运行时肯定的。(这里只是指出来,了解下就行,暂时不必深刻)
我的以为,不必太在乎命名空间和做用域的定义,之因此有命名空间的说法,只是为了引入做用域的概念。
咱们只须要清楚两个方面的内容:一是,哪一个做用域包含哪些名称;二是,相反的,赋值和读值的时候它又是指向哪一个做用域,并理解 nonlocal 和 global 的使用。
https://docs.python.org/3/tutorial/classes.html#python-scopes-and-namespaces
复制代码
本文首发于公众号「小小后端」。