装饰器原理
简单装饰器原理
先写个非常简单的装饰器:
def w1(main_func):
def outer(request,kargs):
print('before') #这里可实现验证功能
main_func(request,kargs)
print('after') #这里可以实现如果main_func不能成功执行之后的操作
return main_func(request, kargs)
return outer
@w1
def show(request,kargs):
print('show')
return True
t = show(1,2)
print(t)
简单装饰器的执行过程:
第一步:先将w1函数加载到内存
第二步:@w1
@函数名是python的一种语法糖
从表面上看,Python会解释这两句代码,因为函数在调用之前内部代码不会被执行。
@w1内部会执行以下操作:
- 将被装饰的函数当做参数传给w1,即w1等价于w1(show)
内部就会执行下边这些代码
def outer(request,kargs):
print('before') #这里可实现验证功能
main_func(request,kargs)
print('after') #这里可以实现如果main_func不能成功执行之后的操作
return main_func(request, kargs)
return outer
然后将执行完的w1函数的返回值赋值给新的show函数,即:
即新show为:
def outer(request,kargs):
print('before')
main_func(request,kargs)
print('after')
return main_func(request, kargs)
所以有人想执行show函数时,就会执行新的show函数,这时会先执行print(‘before’)(如果这里为验证的话会先执行验证),接下来执行原来的show函数,再执行print(‘after’),最后将原show函数的返回值返回给调用者
这样一来,既可以实现验证功能,又可以执行原来show()函数的内容
带参数的复杂装饰器原理
def Login(request,kargs):
print('before')
return 33
def ErrorHandle(request,kargs):
print('after')
def Filter(before_func,after_func):
def outer(main_func):
def wrapper(request,kargs):
before_result = before_func(request,kargs)
main_result = main_func(request, kargs)
after_result = after_func(request, kargs)
return wrapper
return outer
@Filter(Login,ErrorHandle)
def Index(request,kargs):
print('index')
return "home page..."
Index('req','alex')
复杂装饰器的执行过程:
- 首先将Login,ErrorHandle,Filter三个函数写进内存
- @Filter(Login,ErrorHandle),
这里实现了如下功能;
将Login,ErrorHandle当做参数传给Filter函数
接下来将被装饰的Index函数当做参数传给outer,即outer(Index)
内部就会执行下边这些代码;
def wrapper(request,kargs):
before_result = before_func(request,kargs)
main_result = main_func(request, kargs)
after_result = after_func(request, kargs)
return wrapper
然后将执行完的outer函数的返回值赋值给新Index:
即新Index为:
def wrapper(request,kargs):
before_result = before_func(request,kargs)
main_result = main_func(request, kargs)
after_result = after_func(request, kargs)
- 接下来会执行Index(‘req’,‘alex’)
这时的Index已经为新Index
所以接下来会执行Login,原来的Index,
即
def Index(request,kargs):
print('index')
return "home page..."
最后执行ErrorHandle函数
总结
带参数的复杂装饰器可以提高代码的重用率
再执行原函数之前想进行什么操作直接定义一个函数就可以,不必再写一个装饰器
降低了搬砖工人的成本哈