考虑以下装饰器函数,它可以返回已经被装饰的函数,也可以返回带参数的装饰器函数:
from functools import wraps, partial, update_wrapper
from inspect import signature
def wrapit(func=None, *, verb='calling'):
if func is None: # return a decoratOR
return partial(wrapit, verb=verb)
else: # return a decoratED
@wraps(func)
def _func(*args, **kwargs):
print(f'{verb} {func.__name__} with {args} and {kwargs}')
return func(*args, **kwargs)
return _func
示例:
>>> f = lambda x, y=1: x + y
>>> ff = wrapit(verb='launching')(f)
>>> assert ff(10) == 11
launching <lambda> with (10,) and {}
>>> assert signature(ff) == signature(f)
>>>
>>> # but can also use it as a "decorator factory"
>>> @wrapit(verb='calling')
... def f(x, y=1):
... return x + y
...
>>> assert ff(10) == 11
launching <lambda> with (10,) and {}
>>> assert signature(ff) == signature(f)
表单类的外观可能如下所示:
class Wrapit:
def __init__(self, func, verb='calling'):
self.func, self.verb = func, verb
update_wrapper(self, func)
def __call__(self, *args, **kwargs):
print(f'{self.verb} {self.func.__name__} with {args} and {kwargs}')
return self.func(*args, **kwargs)
但是我们如何让类能够以函数式实现的“装饰器工厂”模式运作(由if func is None: return partial...
实现)?
我们如何将这个技巧整合到一个装饰器类中?
__new__
中使用相同的技巧。 - juanpa.arrivillaga__new__
只接受一个类作为参数,所以我怎么可以基于func is None
条件来决定我的操作呢? - thorwhalen