函数作用域:
local--函数内部作用域
enclosing--函数内部与内嵌函数之间
global--全局作用域
build-in--内置作用域
python查找一个变量的顺序:local、enclosing、gloabl、build-in
闭包:
在学习闭包的过程中,找到了很多的描述来理解这个名词,最终自我总结为:
闭包是引用外部作用域(enclosing作用域)变量的内部函数,并且被外部函数返回。
如果是闭包,外部函数在结束运行之后会将变量交给内部函数(将变量添加到了内部函数的属性当中),然后释放内存
# 外部函数
def demo():
# 外部函数的变量
a = 5
# 内部函数
def dem(i):
# 引用了外部函数的变量a
b = a + i
return b
# 被外部函数返回
return dem
b = demo()
print(b(2))
闭包的作用:封装,代码复用
闭包的注意事情:
闭包中不能修改外部函数的局部变量
以下是一个错误示例
def demo():
a = 5
def dem(i):
a = a + i
return b
return dem
b = demo()
b(5)
>>>
Traceback (most recent call last):
File "r21.py", line 41, in <module>
b(5)
File "r21.py", line 36, in dem
a = a + i
UnboundLocalError: local variable 'a' referenced before assignment
出现报错:UnboundLocalError: local variable 'a' referenced before assignment
这是由于Python规则指定在赋值语句左边的变量为函数的局部变量,dem()函数中a被认为是局部变量,所有右边的a python找不到
装饰器:
用来装饰函数,返回一个新的函数对象,被装饰函数标示符指向返回的函数对象
可以理解为装饰器就是对闭包的使用
def demo(func):
print('demo')
print(func)
def in_demo(x,y):
print('in_demo')
print(in_demo)
return func(x,y)
# 第二步,demo返回一个新函数,for_demo指向了in_demo
return in_demo
# 第一步,将for_demo()作为参数传给了demo()
@demo
def for_demo(x, y):
print('for_demo')
print(for_demo)
print(x**y)
# 第三步,调用for_demo其实是调用in_demo,而in_demo的参数是原来的for_demo
for_demo(2, 3)
使用装饰器可以不用更改原函数的代码就可以给函数增加新的功能