Python 基础(十八):命名空间 & 作用域

1 命名空间

1.1 概念

命名空间(namespace)是名称到对象的映射,当前大部分命名空间都是通过 Python 字典来实现的,它的主要作用是避免项目中的名字冲突,每一个命名空间都是相对独立的,在不同的命名空间中可以同名,在相同的命名空间中不可以同名。

1.2 种类

命名空间主要有以下三种:

  • 内置:主要用来存放内置函数、异常等,比如:abs 函数、BaseException 异常。
  • 全局:指在模块中定义的名称,比如:类、函数等。
  • 局部:指在函数中定义的名称,比如:函数的参数、在函数中定义的变量等。

1.3 生命周期

通常在不同时刻创建的命名空间拥有不同的生命周期,看一下三种命名空间的生命周期:

  • 内置:在 Python 解释器启动时创建,退出时销毁。
  • 全局:在模块定义被读入时创建,在 Python 解释器退出时销毁。
  • 局部:对于类,在 Python 解释器读到类定义时创建,类定义结束后销毁;对于函数,在函数被调用时创建,函数执行完成或出现未捕获的异常时销毁。
2 作用域

2.1 概念

作用域是 Python 程序可以直接访问命名空间的文本区域(代码区域),名称的非限定引用会尝试在命名空间中查找名称,作用域是静态的,命名空间是随着解释器的执行动态产生的,因此在作用域中访问命名空间中的名字具有了动态性,即作用域被静态确定,被动态使用。

2.2 种类

Python 有如下四种作用域:

  • 局部:最先被搜索的最内部作用域,包含局部名称。
  • 嵌套:根据嵌套层次由内向外搜索,包含非全局、非局部名称。
  • 全局:倒数第二个被搜索,包含当前模块的全局名称。
  • 内建:最后被搜索,包含内置名称的命名空间。

作用域的搜索顺序通过下图直观的来看一下:

Python 基础(十八):命名空间 & 作用域

Python 中会按上图所示作用域由内向外去搜索名字。

再通过具体代码来对作用域作进一步了解,如下所示:

  •  
# 全局作用域g = 1def outer():    # 嵌套作用域    e = 2    def inner():        # 局部作用域        i = 3

2.3 global & nonlocal

我们先来看一下全局变量与局部变量。

  • 全局变量:定义在函数外部的变量。
  • 局部变量:定义在函数内部的变量。

全局变量可以在整个程序范围内进行访问,而局部变量只能在函数内部访问。通过具体示例看一下:

  •  
# 全局变量d = 0def sub(a, b):    # d 在这为局部变量    d = a - b    print('函数内 : ', d)
sub(9, 1)print('函数外 : ', d)

执行结果:

  •  
函数内 :  8函数外 :  0

当内部作用域想要修改外部作用域的变量时,就要用到 global 和 nonlocal 关键字了,下面通过具体示例来了解一下。

如果我们想将上面示例中 sub() 函数中的 d 变量修改为全局变量,则需使用 global 关键字,示例如下所示:

  •  
# 全局变量d = 0def sub(a, b):    # 使用 global 声明 d 为全局变量    global d    d = a - b    print('函数内 : ', d)
sub(9, 1)print('函数外 : ', d)

执行结果:

  •  
函数内 :  8函数外 :  8

如果需要修改嵌套作用域中的变量,则需用到 nonlocal 关键字。

不使用 nonlocal

我们先来看一下不使用 nonlocal 关键字的执行情况,如下所示:

  •  
def outer():    d = 1    def inner():        d = 2        print('inner:', d)    inner()    print('outer:', d)outer()

执行结果:

  •  
inner:2outer:1

使用 nonlocal

再来看一下使用了 nonlocal 关键字的执行情况,如下所示:

  •  
def outer():    d = 1    def inner():        nonlocal d        d = 2        print('inner:', d)    inner()    print('outer:', d)outer()

执行结果:

  •  
inner:2outer:2

 

Python 基础(十八):命名空间 & 作用域

 



上一篇:undefined和null区别


下一篇:Java内部类