闭包与装饰器
1.闭包:直接与间接修改。
拓展:1T=1024GB 但是,我们实际使用时,可能只有1000GB
类比:我们的电脑硬盘容量不够,需要在加一个硬盘——>直接修改了。
如果插入U盘则是间接修改。
所以,在我们日常编程的开发中,也是需要对已编写的源代码进行增加和修改。开发时,我们一般不动里面的源代码,而是在外部通过装饰器进行增加和修改,而闭包与装饰器本身就是一个嵌套函数。
2.初识闭包函数:
1函数对象:可以作为一个参数进行传递。
2.函数嵌套:函数中还有一个函数。
3.作用域:闭包里面。
闭:封闭:闭包函数就是嵌套函数内部的内层函数。
包:用内层函数来操作外部函数的数据。
作用:通过间接的方式,给函数传入参数和进行操作。
闭包一般是通过装饰器进行操作,单纯的闭包没有必要,都是要配合装饰器进行使用。
---------------------------------------------------------------------------------------------------------------------------------
格式:
def 外部函数名():
def 内部函数名():
操作外部函数代码块
return 内部函数名
外部函数名():
为什么inner后面不加括号呢?如果加了括号,就是调用,是递归的内容。
注意:如果有嵌套函数,而且内层函数是作为一个返回(return)自己的本身,那么就是一个闭包。
3.装饰器(作用域:内层函数)
其中:内层函数本身就是一个源代码,然后添加或修改。
定义:1.本身就是一个函数。
2.把函数作为一个参数进行传递。
运用场景:
不修改源代码和调用方式情况下。给函数添加或者修改功能作用在日常工作中,
一般是批主管给些接口或者框架给用,但你没有权限或不能改源代码和调用方式。
我们就要合理运用装饰器进行修改或增加代码了.
注意:是在不破坏源代码的基础上进行增加和修改。
4.函数使用:
4.1定义函数
4.2使用函数
装饰器:在【在不修改函数代码】和不掉用方式,给函数增加或者修改功能。
格式:
def 装饰器(被装饰函数名):
def inner():
添加功能代码:
被装饰器名() #这个可以添加在代码前面的,就看你是否在被装饰器函数运行前后。
return inner
#调用
变量名=装饰器名(被装饰器名())
变量名()
#源代码
def A():
print("源代码")
#装饰器
def func(func1):
def inner():
print("被装饰的函数")
return inner
A1=funcA()
A1()
5.语法糖@的用法
为了简洁语法,直接在被转饰器上面添加@修饰函数,就是默认把装饰器添加上。
#装饰器
def func(func1):
def inner():
#进行源代码
func1()
print("被装饰的函数")
return inner
#源代码(注意:先有装饰器,再有语法糖)
@func
def A():
print("源代码")
其中:A()是调用的意思。
6.待定长参数的装饰器
def 装饰器(被装饰器命名):
def inner(参数1,参数2)
代码块
被装饰器名()
return inner
@装饰器名
def 源代码名(参数1,参数2):
代码块
源代码
例如:
#decorator装饰器
#项目要有提示正在计算和计算结束
def decorator(func1):
def inner(a,b):
print("正在计算中,此过程不消耗流量")
func1(a,b)
print("计算结束")
return inner
@decorator
def add(nmuber1,number2)
result=number1+number2
print(f"结果为{result}")
add(5200000,1314)
7.带不定长参数的装饰器(通用装饰器)
def inner(*args,**kwargs)
代码块
被装饰器名()
return inner
@装饰器名
def 被装饰器名(定长参数或不定长参数或无参数)
代码块
源代码()
总结:
什么情况会用到装饰器
不修改源代码和调用方式的情靓下,进行增加源代码功能
装饰器本质:闭包
在不知道源代码参数个数或者类型的情况下,写一个装饰器用通用装饰器。
def inner (*args,** kwargs)
语法糖 :@装饰器——>装饰器就是为了打补丁的函数。