装饰器的定义
目前创新互联建站已为1000+的企业提供了网站建设、域名、雅安服务器托管、网站托管运营、企业网站设计、西夏网站维护等服务,公司将坚持客户导向、应用为本的策略,正道将秉承"和谐、参与、激情"的文化,与客户和合作伙伴齐心协力一起成长,共同发展。
就是给已有函数增加额外功能的函数,它本质上就是一个闭包函数。
装饰器的功能特点:
不修改已有函数的源代码
不修改已有函数的调用方式
给已有函数增加额外的功能
在这里我们使用装饰器实现已有函数执行时间的统计的代码来讲解一些它的原理
方式一:
import time
def show():
start = time.time()
print("show run...")
stop = time.time()
print("一共耗时 %s 秒" % (stop-start))
def count_number():
start = time.time()
s = 0
for i in range(1000000):
s += i
print(s)
stop = time.time()
print("一共耗时 %s 秒" % (stop-start))
show()
count_number()
方式一的代码虽然可以实现功能,但我们会发现代码太冗余,定义的两个函数都要写一遍计时功能,代码的重用性不高,所以这方法不好.
方式二:
import time
def count_time(func):
start = time.time()
func()
stop = time.time()
print("一共耗时 %s 秒" % (stop-start))
def show():
print("show run...")
def count_number():
s = 0
for i in range(1000000):
s += i
print(s)
count_time(show)
count_time(count_number)
方式二 封装了一个计时的函数,代码量虽然减少了很多,但是你会发现,我们在调用函数的时候,调用方式不一样,所以这个方法也不好,它修改了函数的使用方式,如果在实际开发中,你不需要计时功能了.那你还需要改,就很麻烦.
方式三:
def count_time(func):
def inner():
start = time.time()
func()
stop = time.time()
print("一共耗时 %s 秒" % (stop-start))
return inner
def show():
print("show run...")
def count_number():
s = 0
for i in range(1000000):
s += i
print(s)
# 这句话就是装饰器的原理
show = count_time(show)
# count_time(show) 单看这就是一个函数调用,
# 我们把show传给了count_time也就是show = func
# 所以count_time 函数中调用的 func()就相当于调用
# show(),因此这个闭包既帮我们完成了计时又完成了show
# 函数的调用,下面的count_number是一样的原理
count_number = count_time(count_number)
show()
count_number()
show = count_time(show)
count_number = count_time(count_number)
以上的代码,是装饰器的原理.我们必须能够理解,这是把 show 函数在内存当中的引用给了count_time
方式三的简写:郑州妇科医院 http://www.sptdfk.com/
如果有多个函数都需要添加登录验证的功能,每次都需要编写show = count_time(show)这样代码对已有函数进行装饰,这种做法还是比较麻烦。
Python给提供了一个装饰函数更加简单的写法,那就是语法糖,语法糖的书写格式是: @装饰器名字,通过语法糖的方式也可以完成对已有函数的装饰
def count_time(func):
def inner():
start = time.time()
func()
stop = time.time()
print("一共耗时 %s 秒" % (stop-start))
return inner
# 使用语法糖方式来装饰函数
@count_time # 等价于 show = count_time(show)
def show():
print("show run...")
@count_time
# 等价于 count_number = count_time(count_number)
def count_number():
s = 0
for i in range(1000000):
s += i
print(s)
show()
count_number()
装饰器的执行时间是加载模块时立即执行
我们需要注意:
闭包函数有且只有一个参数,必须是函数类型,这样定义的函 数才是装饰器。
写代码要遵循开放封闭原则,它规定已经实现的功能代码不允许被修改,但可以被扩展。