一、命名空间
1、定义
名称到对象的映射。命名空间是一个字典的实现,键为变量名,值是变量对应的值。各个命名空间是独立没有关系的,一个命名空间中不能有重名,但是不同的命名空间可以重名而没有任何影响。
2、命名空间的分类
1)、全局命名空间( Global):每个模块加载执行时创建的,记录了模块中定义的变量,包括模块中定义的函数、类、其他导入的模块、模块级的变量与常量。
2)、局部命名空间(Local):每个函数所拥有的命名空间,记录了函数中定义的所有变量,包括函数的入参、内部定义的局部变量。
3)、python内置命名空间(Built-in):任何模块均可以访问,放着内置的函数和异常(比如:input,print,str,list,tuple...)。
3、生命周期
3.1、加载顺序
内置命名空间(程序运行前加载)-->全局命名空间(程序运行中:从上到下加载)-->局部命名空间(程序运行中:调用时才加载)
3.2、取值顺序
在局部调用:局部命名空间-->全局命名空间-->内置命名空间
在全局调用:全局命名空间-->内置命名空间
综上所述,在寻找变量时,从小范围,一层一层到大范围去找寻。
4、作用域
4.1、定义
作用域是针对变量而言,指申明的变量在程序里的可应用范围。或者称为变量的可见性。
4.2、分类
按照生效范围可以分为全局作用域和局部作用域。
1)、全局作用域:包含内置名称空间、全局名称空间,在整个文件的任意位置都能被引用、全局有效
2)、局部作用域:局部名称空间,只能在局部范围内生效
4.3、两个常用的内置函数
locals() 和globals(),它们提供了基于字典的访问局部和全局变量的方式。
1)、locals():函数会以字典类型返回当前位置的全部局部变量。
2)、globals():函数会以字典类型返回当前位置的全部全局变量。
#在全局作用域内
print(globals())
print(locals())
#输出结果:
{'__name__': '__main__', '__doc__': None, '__package__': None, '__loader__':
<_frozen_importlib_external.SourceFileLoader object at 0x0000018DA3453FD0>, '__spec__': None, '__annotations__': {},
'__builtins__': <module 'builtins' (built-in)>, '__file__': 'd:/python10/day10/homework.py', '__cached__': None}
{'__name__': '__main__', '__doc__': None, '__package__': None, '__loader__':
<_frozen_importlib_external.SourceFileLoader object at 0x0000018DA3453FD0>, '__spec__': None, '__annotations__': {},
'__builtins__': <module 'builtins' (built-in)>, '__file__': 'd:/python10/day10/homework.py', '__cached__': None}
#在函数局部作用域内
name = 'wusir'
def func():
name = 'alex'
print(globals())
print(locals()) #{'name': 'alex'}
func()
#输出结果:
{'__name__': '__main__', '__doc__': None, '__package__': None, '__loader__':
<_frozen_importlib_external.SourceFileLoader object at 0x000002397E843FD0>, '__spec__': None, '__annotations__': {},
'__builtins__': <module 'builtins' (built-in)>, '__file__': 'd:/python10/day10/homework.py', '__cached__': None, 'name': 'wusir',
'func': <function func at 0x000002397CA62E18>}
{'name': 'alex'}
5、关键字(global 与 nonlocal)
5.1、global的作用
1)、声明一个全局变量
2)、在局部作用域想要对全局作用域的全局变量进行修改时,需要用到 global(限于字符串,数字)
def func():
global a
a = 2
print(a) #2
func()
print(a) #2
#输出结果
2 2
注意:对可变数据类型(list,dict,set)可以直接引用不用通过global(以下是实例)
li = [1,2,3]
dic = {'a':'b'}
def change():
li.append('a')
dic['q'] = 'g'
print(dic) #{'a': 'b', 'q': 'g'}
print(li) #[1, 2, 3, 'a']
change()
print(li) #[1, 2, 3, 'a']
print(dic) #{'a': 'b', 'q': 'g'}
#输出结果
{'a': 'b', 'q': 'g'}
[1, 2, 3, 'a']
[1, 2, 3, 'a']
{'a': 'b', 'q': 'g'}
5.2、nonlocal的作用
1)、不能修改全局变量
2)、在局部作用域中,对父级作用域(或者更外层作用域非全局作用域)的变量进行引用和修改,并且引用的哪层,从那层及以下此变量全部发生改变
def add_b():
b = 42
def do_global():
b = 10
print(b) #10
def dd_nonlocal():
nonlocal b
b = b + 20
print(b) #30
dd_nonlocal()
print(b) #30
do_global()
print(b) #42
add_b()
#输出结果
10 30 30 42
二、函数的嵌套
def fun1():
print(111)
def fun2():
print(222)
fun1()
fun2()
#输出结果
222
111
函数的作用域链:小范围作用域可以使用大范围的变量,但是反之不行,它是单向的。
print(111) #执行顺序--1
def fun2():
print(222) #执行顺序--3
def fun3():
print(666) #执行顺序--5
print(444) #执行顺序--4
fun3()
print(888) #执行顺序--6
print(333) #执行顺序--2
fun2()
print(555) #执行顺序--7
#输出结果
111 333 222 444 666 888 555