无参装饰器的实现
1.1 双层语法糖
import time
def outer(func):
def get_time(*args, **kwargs):
start_time = time.time()
res = func(*args, **kwargs) # 只能够统计index函数的时间
end_time = time.time()
print('执行时间:%s' % (end_time - start_time))
return res
return get_time
def login_auth(func):
# func = index
def auth():
username = input('username:>>>').strip()
password = input('password:>>>').strip()
# 2. 比较用户名和密码
if username == 'jerry' and password == '123':
# 执行函数
print('登录成功')
func()
else:
print('用户名或者密码错误')
return auth
@login_auth # index=login_auth(get_time) # index=auth
@outer # get_time=outer(index)
def index():
time.sleep(3)
print('from index')
index() # auth()
先执行 @outer
再执行 @login_auth
1.2 三层语法糖(多层)
# 判断七句print执行顺序
def outter1(func1):
print('加载了outter1')
def wrapper1(*args, **kwargs):
print('执行了wrapper1')
res1 = func1(*args, **kwargs)
return res1
return wrapper1
def outter2(func2):
print('加载了outter2')
def wrapper2(*args, **kwargs):
print('执行了wrapper2')
res2 = func2(*args, **kwargs)
return res2
return wrapper2
def outter3(func3):
print('加载了outter3')
def wrapper3(*args, **kwargs):
print('执行了wrapper3')
res3 = func3(*args, **kwargs)
return res3
return wrapper3
@outter1
@outter2
@outter3
def index():
print('from index')
index()
打印顺序:
加载了outter1
加载了outter2
加载了outter3
执行了wrapper1
执行了wrapper2
执行了wrapper3
from index
1.3 装饰器的修复技术
import time
from functools import wraps
def outer(func):
@wraps(func) # 修复技术
def get_time():
start_time = time.time()
func() # 只能够统计index函数的时间
end_time = time.time()
print('执行时间:%s' % (end_time - start_time))
return get_time
# @outer # index=outer(index)
def index():
print('from index')
'''修复技术就是为了让装饰器伪装的更像'''
有参装饰器的实现
了解无参装饰器的实现原理后,我们可以再实现一个用来为被装饰对象添加认证功能的装饰器,实现的基本形式如下
def deco(func):
def wrapper(*args,**kwargs):
编写基于文件的认证,认证通过则执行res=func(*args,**kwargs),并返回res
return wrapper
如果我们想提供多种不同的认证方式以供选择,单从wrapper函数的实现角度改写如下
def deco(func):
def wrapper(*args,**kwargs):
if driver == 'file':
编写基于文件的认证,认证通过则执行res=func(*args,**kwargs),并返回res
elif driver == 'mysql':
编写基于mysql认证,认证通过则执行res=func(*args,**kwargs),并返回res
return wrapper
函数wrapper需要一个driver参数,而函数deco与wrapper的参数都有其特定的功能,不能用来接受其他类别的参数,可以在deco的外部再包一层函数auth,用来专门接受额外的参数,这样便保证了在auth函数内无论多少层都可以引用到
def auth(driver):
def deco(func):
……
return deco
此时我们就实现了一个有参装饰器,使用方式如下
# 先调用auth_type(driver='file'),得到@deco,deco是一个闭包函数,包含了对外部作用域名字driver的引用,@deco的语法意义与无参装饰器一样
@auth(driver='file')
def index():
pass
@auth(driver='mysql')
def home():
pass