名称空间与作用域
之前在函数的嵌套里遇到过函数内部定义的函数不能在函数外面使用,这是为么呢?
这就涉及到名称空间与作用域了
名称空间
名称空间就是存放名字的空间,比如变量名、函数名
内置名称空间
这是Python解释器自带的,如int、str、len……
,它们在python解释器启动的时候就已经占据了内存空间,当python解释器关闭时才会失效
局部名称空间
用于存放函数调用期间函数体产生的名字,只有在函数调用的时候才会启动,函数执行完成后就失效了
def f1():
def f2():
print('from f2')
f2()
f1()
from f2
全局名称空间
除去内置和局部的名字之外,其余的全部存放在全局名称空间,当py文件执行的时候生效,文件执行结束后失效
x = 10 # 全局名称空间
def f1():
x = 20 # 局部名称空间
def f2():
print('from f1',x)
f2()
f1()
print('from file.py',x)
from f1 20
from file.py 10
加载顺序
当pythn解释器已打开的时候,python自带的内置函数就已经进行加载,然后文件才会打开,此时才会有全局名称空间,当文件里的某个函数被调用时,函数内部的名称才会占用名称空间。
内置名称空间---->全局名称空间---->局部名称空间
查找顺序
y = 20
x = 15
def f1():
x = 3
def f2():
print(x,y)
f2()
f1()
3 20
从上面的函数可以看出,当f2需要打印x,y的值得时候,会优先从里面开始寻找,里面找不到的时候才会向外寻找
从当前位置开始查找,如果当前位置为局部空间,则查找顺序为:局部--》全局--》内置
作用域
作用域就是起作用的区域
全局作用域
即全局有效,对内置名称空间和全局名称空间都有效
x = 1
def f1():
print(x)
f1()
1
局部作用域
即只对局部有用,临时存储,只包含局部名称空间
x = 10 # 全局作用域
def f1():
x = 20 # 局部作用域
print(x)
def f2():
print(x) # 依照就近原则,由内而外寻找
f2()
f1()
print(x)
20
20
10
- 注意点:
作用域在定义的时候就固定死了,与函数的调用无关
x = 1
def f1():
print(x)
def f2():
x = 5
print(x)
f1()
f2()
1
5
函数对象+作用域应用
def f1():
def inner():
print('from inner')
return inner
f = f1() # 把局部定义的函数放在了全局之中
def f2():
f()
f2()
from inner
补充知识点
global关键字
修改全局作用域中的变量
x = 10
def f1():
global x
x = 5
f1()
print(x)
5
x = 1
def f1():
global x
x = 2
print(1,x)
def f2():
global x # 修改全局
x = 3
print(2,x)
f2()
print(3,x)
f1()
print(4,x)
1 2
2 3
3 3
4 3
nonlocal关键字
修改局部作用域中的变量
x = 1
def f1():
x = 2
print(1,x)
def f2():
nonlocal x # 修改局部
x = 3
print(2,x)
f2()
print(3,x)
f1()
print(4,x)
1 2
2 3
3 3
4 1
注意点
- 局部变量想要修改全局变量的可变类型,不需要任何声明,可直接修改
- 在局部如果想要修改全局的不可变类型,需要先借助global声明,声明为全局的变量,即可直接修改。
lis = [1,2,3,4]
def f1():
lis.append('a')
print(f'调用函数前的列表:{lis}')
f1()
print(f'调用函数后的列表:{lis}')
调用函数前的列表:[1, 2, 3, 4]
调用函数后的列表:[1, 2, 3, 4, 'a']