这个概念一开始真的很难理解,以至于现在我去写这篇日志,我都不确定是否真的理解了装饰器的意义。首先,假定我们定义一个函数date()
:
def date():
print("2017/9/17")
这个函数的作用很简单,就是打印日期。但是如果我们现在希望,在正常书写调用date()
函数时,它可以申明自己已经被调用(也就是额外加上一个功能)。
如果不额外强求正常书写:
def call_me(func):
print("{} is running").formate(func))
func()
def date():
print("2017/9/17")
call_me(date)
以上函数就可以实现这个功能。
但是,有个问题。因为我只是希望,在我调用date()
时,可以附带上一些功能,而不需要改写一些代码。上面的实现方法,已经完全看不到直接调用date()
的影子了。这个时候,可以用装饰器实现:
def call_me(func):
def wrapper():
print("{} is running").formate(func))
return func()
return wrapper()
def date():
print("2017/9/17")
date = call_me(date)
date()
这时,我们想使用附带功能的date()
函数,也一样只需要调用date()
,而不是用call_me(date)
。其中call_me()
就是一个装饰器。简单点说,其实,call_me()
是把额外的功能连同date()
本身封装到了一起。call_me()
本身返回了一个函数wrapper()
,语句date = call_me(date)
相当于是date = wrapper
,那么调用了date()
就相当于调用了wrapper()
,而wrapper()
函数就是一个带有额外功能的date()
。
现在最新的实例看起来还不够简洁,因为我们的终极目的是,越简单越好。这时,就要用到@
。新的实例如下:
def call_me(func):
def wrapper():
print("{} is running").formate(func))
return func()
return wrapper()
@call_me
def date():
print("2017/9/17")
date()
这样就是实现了我们的最初目的。现在看起来,call_me
就像是装饰一样在date()
函数前面。运行的结果是这样的:
>>>date is running
2017/9/17
我自己对装饰器的理解比较粗略,也就只能阐述于此,纰漏甚多。
网友评论