python使用命名空间记录变量。python中的命名空间就像是一个dict,key是变量的名字,value是变量的值。

python中,每个函数都有一个自己的命名空间,叫做local namespace,它记录了函数的变量。
python中,每个module有一个自己的命名空间,叫做global namespace,它记录了module的变量,包括 functions, classes 和其它imported modules,还有 module级别的 变量和常量。
还有一个build-in 命名空间,可以被任意模块访问,这个build-in命名空间中包含了build-in function 和 exceptions。

当python中的某段代码要访问一个变量x时,python会在所有的命名空间中寻找这个变量,查找的顺序为:

local namespace - 指的是当前函数或者当前类方法。如果在当前函数中找到了变量,停止搜索
global namespace - 指的是当前的模块。如果在当前模块中找到了变量,停止搜索
build-in namespace - 如果在之前两个namespace中都找不到变量x,python会假设x是build-in的函数或者变量。如果x不是内置函数或者变量,python会报错NameError。

对于闭包来说,这里有一点区别,如果在local namespace中找不到变量的话,还会去父函数的local namespace中找变量。

 

locals

内置函数locals(), 返回当前函数(方法)的局部命名空间

 

def func(a = 1):
    b = 2
    print(locals())
    return a+b
func()
# {'a': 1, 'b': 2} 可以看出,locals返回的是个dict123456

  

globals

内置函数globals(),返回当前module的命名空间

def func(a = 1):
    b = 2
    return a+b
func()
print(globals()) # globals()返回的也是个dict12345

  

 

locals()和globals()有一个区别是,locals只读,globals可以写

 

def func(a = 1):
    b = 2
    return a+b
func()
glos = globals()
glos['new_variable'] = 3
print(new_variable)
# 3  , 我们并没有显示定义new_variable这个变量,只是在globals中添加了这个key,在随后的代码中,
#就可以像访问一般变量一样来访问。
def func(a = 1):
    b = 2
    locs = locals()
    locs['c']  = 1
    print(c)
func()
# NameError: name 'c' is not defined1234567891011121314151617

  

from module import 和 import module

使用import module时,module本身被引入,但是保存它原有的命名空间,所以我们需要使用module.name这种方式访问它的 函数和变量。
from module import这种方式,是将其它模块的函数或者变量引到当前的命名空间中,所以就不需要使用module.name这种方式访问其它的模块的方法了。

 

if __name__ trick

python中的module也是对象,所有的modules都有一个内置的属性__name__,模块的__name__属性的值取决于如何使用这个模块,如果import module,那么__name__属性的值是模块的名字。如果直接执行这个模块的话,那么__name__属性的值就是默认值__main__。

 

module的一些内置属性

__name__: 上面已经介绍过
__file__ : 当前module的绝对路径
__dict__:
__doc__ :
__package__:
__path__: