命名空间(namespaces)与作用域(global、nonlocal)

一、命名空间


命名空间即存放名字与对象映射/绑定关系的地方,对栈区的划分。对于x=3,Python会申请内存空间存放对象3,然后将名字x与3的绑定关系存放于命名空间中,del x表示清除该绑定关系。
​ 在程序执行期间最多会存在三种命名空间(相当于把栈区划分了三部分),有了名称空间后,就可以在栈区存放相同的名字了。

在这里插入图片描述

1.1 内建命名空间

存活周期:伴随python解释器启动/关闭而产生/回收,因而是第一个被加载的命名空间,用来存放一些内置的名字,比如内建函数名

>>> max<built-in function max> #built-in内建

1.2 全局命名空间

存放的名字:只要不是函数内定义、也不是内置的,剩下的都是全局名称空间的名字
存活周期:python文件执行则产生,python文件运行完毕后销毁
伴随python文件的开始执行/执行完毕而产生/回收,是第二个被加载的命名空间,文件执行过程中产生的名字都会存放于该命名空间中,如下名字

import sys #模块名sys
x=1 #变量名x
if x == 1:
    y=2 #变量名y
def foo(x): #函数名foo
    y=1 # 此y不是
    def bar(): # 此bar不是
        pass
Class Bar: #类名Bar
    pass

1.3 局部命名空间

存放的名字:在调用函数时,运行函数体代码过程中产生的函数内的名字
存活周期:在调用函数时存活,函数调用完毕后则销毁
伴随函数的调用/结束而临时产生/回收,函数的形参、函数内定义的名字都会被存放于该命名空间中
重点: 局部命名空间不一定有 ,即使同一个函数调用多次,也会产生多个局部命名空间,调用完销毁。因为要保证彼此独立

def foo(x):
    y=3 #调用函数时,才会执行函数代码,名字x和y都存放于该函数的局部命名空间中

1.4命名空间的加载顺序:

内置命名空间->全局命名空间->局部命名空间

1.5查找顺序:

而查找一个名字,必须从三个命名空间之一找到,查找顺序为:局部命名空间->全局命名空间->内置命名空间。

1.6 销毁顺序:局部名称空间>全局名空间>内置名称空间

示范1:

def func():
    print(x)
x=111

func()  # 111

示范2:名称空间的"嵌套"关系是以函数定义阶段为准,与调用位置无关(重点掌握)

# x=1 	
# def func():
#    print(x)
#
#
# def foo():
#     x=222
#     func() 
#
# foo() # 结果为1

师范3:

 # 示范3:函数嵌套定义
input=111
def f1():
    def f2():
        # input=333
        print(input)
    input=222
    f2()
f1() #222

# 示范4:错误示范
# x=111
# def func():
#     print(x) 
#     x=222
# func()

二、作用域


2.1 全局作用域与局部作用域

按照名字作用范围的不同可以将三个命名空间划分为两个区域:

  1. 全局作用域:位于全局命名空间、内建命名空间中的名字属于全局范围,该范围内的名字全局存活(除非被删除,否则在整个文件执行过程中存活)、全局有效(在任意位置都可以使用);
  2. 局部作用域:位于局部命名空间中的名字属于局部范围。该范围内的名字临时存活(即在函数调用时临时生成,函数调用结束后就释放)、局部有效(只能在函数内使用)。

2.2 作用域与名字查找的优先级

​在局部作用域查找名字时,起始位置是局部作用域,所以先查找局部命名空间,没有找到,再去全局作用域查找:先查找全局命名空间,没有找到,再查找内置命名空间,最后都没有找到就会抛出异常

x=100 #全局作用域的名字x
def foo():
    x=300 #局部作用域的名字x
    print(x) #在局部找x
foo()#结果为300

# LEGB
# # built-in
# # global
# def f1():
#     # enclosing
#     def f2():
#         # enclosing
#         def f3():
#             # local
#             pass

在全局作用域查找名字时,起始位置便是全局作用域,所以先查找全局命名空间,没有找到,再查找内置命名空间,最后都没有找到就会抛出异常

x=100
def foo():
    x=300 #在函数调用时产生局部作用域的名字x
foo()
print(x) #在全局找x,结果为100

提示:可以调用内建函数locals()和globals()来分别查看局部作用域和全局作用域的名字,查看的结果都是字典格式。在全局作用域查看到的locals()的结果等于globals()

Python支持函数的嵌套定义,在内嵌的函数内查找名字时,会优先查找自己局部作用域的名字,然后由内而外一层层查找外部嵌套函数定义的作用域,没有找到,则查找全局作用域

x=1
def outer():
    x=2
    def inner(): # 函数名inner属于outer这一层作用域的名字
        x=3
        print('inner x:%s' %x)

    inner()
    print('outer x:%s' %x)

outer() 
#结果为
inner x:3
outer x:2

在函数内,无论嵌套多少层,都可以查看到全局作用域的名字,若要在函数内修改全局命名空间中名字的值,当值为不可变类型时,则需要用到global关键字

x=1
def foo():
    global x #声明x为全局名称空间的名字
    x=2
foo()
print(x) #结果为2

当实参的值为可变类型时,函数体内对该值的修改将直接反应到原值,
num_list=[1,2,3]
def foo(nums):
    nums.append(5)

foo(num_list)
print(num_list)
#结果为
[1, 2, 3, 5]

对于嵌套多层的函数,使用nonlocal关键字可以将名字声明为来自外部嵌套函数定义的作用域(非全局)
也就是当你想改变函数外嵌套的函数的不可变类型名字对应的值

def  f1():
    x=2
    def f2():
        nonlocal x
        x=3
    f2() #调用f2(),修改f1作用域中名字x的值
    print(x) #在f1作用域查看x

f1()

#结果 3


案例2
x=0
def f1():
    x=11
    def f2():
        x=22
        def f3():
            nonlocal x  # 永远修改的是它的相邻外层
            x=33
        f3()
        print('f2里面的x',x)
    f2()
    print('f1内的x:',x)

f1()

# 结果
# f2里面的x 33
# f1内的x: 11
# 全局x 0

nonlocal x会从当前函数的外层函数开始一层层去查找名字x,若是一直到最外层函数都找不到,则会抛出异常。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值