装饰器是一个返回函数的函数。写一个装饰器,除了最常见的在函数中定义函数以外,python还允许使用类来定义一个装饰器。
ps注意:不管你是为了python就业还是兴趣爱好,记住:项目开发经验永远是核心,如果你缺新项目练习或者没有python精讲教程,可以去小编的python交流.裙 :七衣衣九七七巴而五(数字的谐音)转换下可以找到了,里面很多新教程项目,还可以跟老司机交流讨教!
1、用类写装饰器
下面用常见的写法实现了一个缓存装饰器。
def cache(func):data = {}def wrapper(*args, **kwargs):key = f'{func.__name__}-{str(args)}-{str(kwargs)})'ifkeyindata:result = data.get(key) print('cached')else:result = func(*args, **kwargs)data[key] = result print('calculated')returnresultreturn wrapper看看缓存的效果。
@cachedef rectangle_area(length, width):return length * widthrectangle_area(2, 3)# calculated# 6rectangle_area(2, 3)# cached# 6装饰器的@cache是一个语法糖,相当于func = cache(func),如果这里的cache不是一个函数,而是一个类又会怎样呢?定义一个类class cache, 那么调用func = cache(func)会得到一个对象,这时返回的func其实是cache的对象。定义__call__方法可以将类的实例变成可调用对象,可以像调用函数一样调用对象。然后在__call__方法里调用原本的func函数就能实现装饰器了。所以cache类也能当作装饰器使用,并且能以@cache的形式使用。
接下来把cache函数改写为cache类:
class cache:def __init__(self, func): self.func = func self.data = {}def __call__(self, *args, **kwargs): func = self.funcdata = self.datakey = f'{func.__name__}-{str(args)}-{str(kwargs)})'ifkeyindata:result = data.get(key) print('cached')else:result = func(*args, **kwargs)data[key] = result print('calculated')returnresult再看看缓存结果,效果一样。
@cachedef rectangle_area(length, width):return length * widthrectangle_area(2, 3)# calculated# 6rectangle_area(2, 3)# cached# 62、装饰类的方法
装饰器不止能装饰函数,也经常用来装饰类的方法,但是我发现用类写的装饰器不能直接用在装饰类的方法上。(有点绕…)
先看看函数写的装饰器如何装饰类的方法。
class rectangle:def __init__(self, length, width):self.length = lengthself.width = width @cachedef area(self):returnself.length * self.widthr = rectangle(2, 3)r.area()# calculated# 6r.area()# cached# 6但是如果直接换成cache类会报错,这个错误的原因是area被装饰后变成了类的一个属性,而不是方法。
class rectangle:def __init__(self, length, width):self.length = lengthself.width = width @cachedef area(self):returnself.length * self.widthr = rectangle(2, 3)r.area()# typeerror: area() missing 1 required positional argument: 'self'rectangle.area# <__main__.cache object at 0x0000012d8e7a6d30>r.area# <__main__.cache object at 0x0000012d8e7a6d30>回头再来看看没有装饰器的情况,python在实例化对象后把函数变成了方法。
class rectangle:def __init__(self, length, width):self.length = lengthself.width = widthdef area(self):returnself.length * self.widthrectangle.area# <function rectangle.area at 0x0000012d8e7b28c8>r = rectangle(2, 3)r.area# <bound method rectangle.area of <__main__.rectangle object因此解决办法很简单,要用类写的装饰器来装饰类的方法,只需要把可调用对象包装成函数就行。
# 定义一个简单的装饰器,什么也不做,仅仅是把可调用对象包装成函数def method(call):def wrapper(*args, **kwargs):return call(*args, **kwargs)return wrapperclass rectangle:def __init__(self, length, width):self.length = lengthself.width = width @method @cachedef area(self):returnself.length * self.widthr = rectangle(2, 3)r.area()# calculated# 6r.area()# cached# 6或者用@property还能直接把方法变成属性。
class rectangle:def __init__(self, length, width):self.length = lengthself.width = width @property @cachedef area(self):returnself.length * self.widthr = rectangle(2, 3)r.area# calculated# 6r.area# cached# 6总结
用类写装饰器并非什么特别的技巧,一般情况下确实没必要这么写,不过这样就可以用一些类的特性来写装饰器,比如类的继承,也算是提供了另一种思路吧。
最后注意:不管你是为了python就业还是兴趣爱好,记住:项目开发经验永远是核心,如果你缺新项目练习或者没有python精讲教程,可以去小编的python交流.裙 :七衣衣九七七巴而五(数字的谐音)转换下可以找到了,里面很多新教程项目,还可以跟老司机交流讨教!
本文的文字及图片来源于网络加上自己的想法,仅供学习、交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理。