前记
本质:装饰器的本质是函数,也是用def去定义的;
功能:装饰其他函数,即:为其他函数添加附加功能;
原则:不修改被装饰的函数的源代码,不能修改被装饰的函数的调用方式;示例:
import time
def timer(func):
def warp(*args,**kw):
start_time = time.time()
func()
end_time = time.time()
print("the func run time is %s",(end_time - start_time))
return warp@timer
def test1():
time.sleep(3)
print("in the test1")if __name__ == "__main__":
test1()
输出:
in the test1
the func run time is %s 3.0010013580322266
结论:统计函数运行时间的装饰器;
一、变量调用函数
示例:
def log(func):
def warp(*args,**kw):
print("call %s():",func.__name__)
return func(*args,**kw)
return warp@log
def now():
print("2018-08-08")f = now
f()
print(now.__name__)
print(f.__name__)
输出:
call %s(): now
2018-08-08
warp
warp
结论:
1、函数对象可复制给变量,通过变量调用函数;
2、函数对象__name__属性,可获取函数名字;
3、增强now函数功能,在函数前后打印日志,又不修改now()函数定义,这种代码运行期间动态增加功能的方式,称之为装饰器Decorator;
4、本质上,decorator是一个返回函数的高阶函数;
5、log装饰器,接收一个函数作为参数,并返回一个函数,使用@语法,把decorator置于函数定义处;
6、调用now函数,不仅返回now函数本身,还会在now函数前打印一行日志;
7、把@log放到now()函数定义处,相当于执行了now = log(now),log是装饰器,返回一个函数,所以原now函数仍然存在,现在同名now函数指向新的函数,于是调用now函数将执行新函数,即在log函数中返回的warp()函数;
8、warp()函数,先打印日志,接着调用原函数;
9、now.__name__变为warp,因为返回的函数名就是warp,所以要把原函数的__name__复制到warp中,使用functools.warps。
示例:
import functoolsdef log(func):
@functools.wraps(func)
def warp(*args,**kw):
print("call %s():",func.__name__)
return func(*args,**kw)
return warp@log
def now():
print("2018-08-08")f = now
f()
print(now.__name__)
print(f.__name__)
输出:
call %s(): now
2018-08-08
now
now
二、小结
1、OOP的设计模式中,decorator被称为装饰模式,装饰模式需要通过继承和组合实现;
2、python的decorator可用函数实现,也可用类实现;
注:如有错误,请指正.