本篇文章给大家带来了关于Python的相关知识,其中主要介绍了关于装饰器的相关问题,包括了闭包、装饰器、使用多个装饰器、带参数的装饰器等等内容,下面一起来看一下,希望对大家有帮助。
|
本篇文章给大家带来了关于python的相关知识,其中主要介绍了关于装饰器的相关问题,包括了闭包、装饰器、使用多个装饰器、带参数的装饰器等等内容,下面一起来看一下,希望对大家有帮助。
推荐学习:python视频教程 一、闭包要了解什么是装饰器(decorator),我们首先需要知道闭包(closure)的概念。 闭包,又称闭包函数或者闭合函数,通俗一点来讲,当某个函数被当成对象返回时还夹带了外部变量,就形成了一个闭包。 以打印Hello World为例,我们先来看一下嵌套函数的结构应该是什么样的: def print_msg(msg):
def printer():
print(msg)
printer()print_msg('Hello World')# Hello World执行 我们再来看一下如果是闭包,该是什么样的结构: def print_msg(msg):
def printer():
print(msg)
return printer
my_msg = print_msg('Hello World')my_msg()# Hello World
执行 def printer():
print('Hello World')于是调用 那么如何判断一个函数是否是闭包函数呢?闭包函数的 def outer(content):
def inner():
print(content)
return innerprint(outer.__closure__)
# Noneinner = outer('Hello World')print(inner.__closure__)
# (<cell at 0x0000023FB1FD0B80: str object at 0x0000023FB1DC84F0>,)由此可见 我们还可以查看闭包所携带的外部变量: print(inner.__closure__[0].cell_contents)# Hello World 说了那么多,那么闭包究竟有什么用呢?闭包存在的意义就是它夹带了外部变量(私货),如果它不夹带私货,那么就和普通的函数没有任何区别。 闭包的优点如下:
二、装饰器我们先考虑这样一个场景,假设先前编写的一个函数已经实现了4个功能,为简便起见,我们用 def module():
print('功能1')
print('功能2')
print('功能3')
print('功能4')现在,由于某种原因,你需要为 def module():
print('功能1')
print('功能2')
print('功能3')
print('功能4')
print('功能5')但在现实业务中,直接做出这样的修改往往是比较危险的(会变得不易于维护)。那么如何在不修改原函数的基础上去为它新添一个功能呢? 你可能已经想到了使用之前的闭包知识: def func_5(original_module):
def wrapper():
original_module()
print('功能5')
return wrapper
new_module = func_5(module)new_module()# 功能1# 功能2# 功能3# 功能4# 功能5 可以看出,我们的新模块:
当然,Python有更简洁的写法(称之为语法糖),我们可以将@符号与装饰器函数的名称一起使用,并将其放置在要装饰的函数的定义上方: def func_5(original_module):
def wrapper():
original_module()
print('功能5')
return wrapper@func_5def module():
print('功能1')
print('功能2')
print('功能3')
print('功能4')module()# 功能1# 功能2# 功能3# 功能4# 功能5基于此,我们可以在不修改原函数的基础上完成计时任务(计算原函数的运行时间),如下: def timer(func):
def wrapper():
import time
tic = time.time()
func()
toc = time.time()
print('程序用时: {}s'.format(toc - tic))
return wrapper@timerdef make_list():
return [i * i for i in range(10**7)]my_list = make_list()# 程序用时: 0.8369960784912109s事实上, def wrapper():
import time
tic = time.time()
a = func()
toc = time.time()
print('程序用时: {}s'.format(toc - tic))
return a三、使用多个装饰器假如我们要为 好在Python允许同时使用多个装饰器: def func_5(original_module):
def wrapper():
original_module()
print('功能5')
return wrapperdef func_6(original_module):
def wrapper():
original_module()
print('功能6')
return wrapper@func_6@func_5def module():
print('功能1')
print('功能2')
print('功能3')
print('功能4')module()# 功能1# 功能2# 功能3# 功能4# 功能5# 功能6上述过程实际上等价于: def module():
print('功能1')
print('功能2')
print('功能3')
print('功能4')new_module = func_6(func_5(module))new_module()此外,需要注意的是,在使用多个装饰器时,最靠近函数定义的装饰器会最先装饰该函数,如果我们改变装饰顺序,则输出结果也将改变: @func_5@func_6def module():
print('功能1')
print('功能2')
print('功能3')
print('功能4')module()# 功能1# 功能2# 功能3# 功能4# 功能6# 功能5四、被装饰的函数带有参数如果被装饰的函数带有参数,那该如何去构造装饰器呢? 考虑这样一个函数: def pide(a, b):
return a / b当b=0 时会出现 因为我们的 def smart_pide(func):
def wrapper(a, b):
if b == 0:
return '被除数不能为0!'
else:
return func(a, b)
return wrapper使用该装饰器进行装饰: @smart_pidedef pide(a, b):
return a / bprint(pide(3, 0))# 被除数不能为0!print(pide(3, 1))# 3.0如果不知道要被装饰的函数有多少个参数,我们可以使用下面更为通用的模板: def decorator(func):
def wrapper(*args, **kwargs):
# ...
res = func(*args, **kwargs)
# ...
return res # 也可以不return
return wrapper五、带参数的装饰器我们之前提到的装饰器都没有带参数,即语法糖
考虑这样一个场景。假如我们在为 def func_5_with_name(name=None):
def func_5(original_module):
def wrapper():
original_module()
print('功能5由{}实现'.format(name))
return wrapper return func_5效果如下: @func_5_with_name(name='若水')def module():
print('功能1')
print('功能2')
print('功能3')
print('功能4')module()# 功能1# 功能2# 功能3# 功能4# 功能5由若水实现对于这种三层嵌套函数,我们可以这样理解:当为 六、使用类作为装饰器将类作为装饰器,我们需要实现 以计时器为例,具体实现如下: class Timer:
def __init__(self, func):
self.func = func def __call__(self):
import time
tic = time.time()
self.func()
toc = time.time()
print('用时: {}s'.format(toc - tic))@Timerdef make_list():
return [i**2 for i in range(10**7)]make_list()# 用时: 2.928966999053955s如果想要自定义生成列表的长度并获得列表(即被装饰的函数带有参数情形),我们就需要在 class Timer:
def __init__(self, func):
self.func = func def __call__(self, num):
import time
tic = time.time()
res = self.func(num)
toc = time.time()
print('用时: {}s'.format(toc - tic))
return res@Timerdef make_list(num):
return [i**2 for i in range(num)]my_list = make_list(10**7)# 用时: 2.8219943046569824sprint(len(my_list))# 10000000如果要构建带参数的类装饰器,则不能把 接下来我们使用类装饰器来复现第五章节中的效果: class Func_5:
def __init__(self, name=None):
self.name = name def __call__(self, func):
def wrapper():
func()
print('功能5由{}实现'.format(self.name))
return wrapper@Func_5('若水')def module():
print('功能1')
print('功能2')
print('功能3')
print('功能4')module()# 功能1# 功能2# 功能3# 功能4# 功能5由若水实现七、内置装饰器Python中有许多内置装饰器,这里仅介绍最常见的三种: 7.1 @classmethod
具体请看下例: class A:
num = 100
def func1(self):
print('功能1')
@classmethod
def func2(cls):
print('功能2')
print(cls.num)
cls().func1()A.func2()# 功能2# 100# 功能17.2 @staticmethod
具体如下: class A:
@staticmethod
def add(a, b):
return a + bprint(A.add(2, 3))# 5print(A().add(2, 3))# 57.3 @property使用 class A:
@property
def printer(self):
print('Hello World')a = A()a.printer# Hello World除此之外, class A:
def __init__(self):
self.name = 'ABC'a = A()print(a.name)# ABCa.name = 1print(a.name)# 1可以看出类中的属性 class A:
def __init__(self):
self.name_ = 'ABC'
@property
def name(self):
return self.name_
a = A()print(a.name)# ABCa.name = 1print(a.name)# AttributeError: can't set attribute推荐学习:python视频教程 以上就是归纳总结Python中的装饰器知识点的详细内容,更多请关注模板之家(www.mb5.com.cn)其它相关文章! |
