Python 中的装饰器类

Posted

技术标签:

【中文标题】Python 中的装饰器类【英文标题】:Decorator classes in Python 【发布时间】:2010-10-14 13:15:09 【问题描述】:

我想按照以下原则构建用作装饰器的类:

    应该可以在 1 个函数的顶部堆叠多个此类装饰器。 生成的函数名称指针应该与没有装饰器的同一个函数无法区分,可能只是因为它是哪个类型/类。 除非装饰器确实强制执行,否则取消装饰器的命令不应该是相关的。 IE。可以按任何顺序应用独立的装饰器。

这是一个 Django 项目,我现在正在处理的具体案例需要 2 个装饰器,并显示为普通的 python 函数:

@AccessCheck
@AutoTemplate
def view(request, item_id) 

@AutoTemplate 更改了函数,因此它不返回 HttpResponse,而是返回一个字典以在上下文中使用。使用了一个RequestContext,模板名是从方法名和模块中推断出来的。

@AccessCheck 根据 item_id 添加对用户的额外检查。

我猜只是为了让构造函数正确并复制适当的属性,但是这些是哪些属性?

下面的装饰器不会像我描述的那样工作:

class NullDecl (object):
    def __init__ (self, func):
        self.func = func
    def __call__ (self, * args):
        return self.func (*args)

如下代码所示:

@NullDecl
@NullDecl
def decorated():
    pass

def pure():
    pass

# results in set(['func_closure', 'func_dict', '__get__', 'func_name',
# 'func_defaults', '__name__', 'func_code', 'func_doc', 'func_globals'])
print set(dir(pure)) - set(dir(decorated));

此外,尝试在 NullDecl 构造函数中添加“print func.name”,它将适用于第一个装饰器,但不适用于第二个 - 因为 name 将丢失。

稍微提炼了eduffy的回答,似乎效果不错:

class NullDecl (object):
    def __init__ (self, func):
        self.func = func
        for n in set(dir(func)) - set(dir(self)):
            setattr(self, n, getattr(func, n))

    def __call__ (self, * args):
        return self.func (*args)
    def __repr__(self):
        return self.func

【问题讨论】:

+1 因为我不确定为什么有人否决了这个问题。这是一个很好的问题:关于 SO 的大多数装饰器问题都不使用类作为装饰器。无论人们如何想到使用类而不是函数,这是一个有效的问题,并且可能完全适合用例。 【参考方案1】:

一个无所事事的装饰器类看起来像这样:

class NullDecl (object):
   def __init__ (self, func):
      self.func = func
      for name in set(dir(func)) - set(dir(self)):
        setattr(self, name, getattr(func, name))

   def __call__ (self, *args):
      return self.func (*args)

然后就可以正常应用了:

@NullDecl
def myFunc (x,y,z):
   return (x+y)/z

【讨论】:

我添加了自己的修复程序,因此它可以正常工作。感谢您的帮助,通常有助于开始非常基本的操作。 也许还有一个地方可以将**kwargs 添加到def __call__ (self, *args): 的定义中(以及稍后对self.func 的调用)? 如何将它与 functool.wraps 一起使用?【参考方案2】:

decorator module 可帮助您编写保留签名的装饰器。

PythonDecoratorLibrary 可能会为装饰器提供有用的示例。

【讨论】:

【参考方案3】:

要创建一个装饰器来包装函数,使其与原始函数无法区分,请使用functools.wraps

例子:


def mydecorator(func):
    @functools.wraps(func):
    def _mydecorator(*args, **kwargs):
        do_something()
        try:
            return func(*args, **kwargs)
        finally:
            clean_up()
    return _mydecorator

# ... and with parameters
def mydecorator(param1, param2):
    def _mydecorator(func):
        @functools.wraps(func)
        def __mydecorator(*args, **kwargs):
            do_something(param1, param2)
            try:
                return func(*args, **kwargs)
            finally:
                clean_up()
        return __mydecorator
    return _mydecorator

(我个人的偏好是使用函数而不是类来创建装饰器)

装饰器的顺序如下:


@d1
@d2
def func():
    pass

# is equivalent to
def func():
    pass

func = d1(d2(func))

【讨论】:

我更喜欢使用类,但我想这是由于我的 Java 背景。我想我读到有人推荐使用类作为装饰器,但我想这是个坏建议。 如果你想使用类,我不确定你把@functools.wraps 放在哪里。我觉得作为装饰器的函数更 Pythonic。而且代码更短。 在您的参数示例中,“func”来自哪里?它不在 mydecorator() 的参数列表中... You apply @decorator(param1, param2) def func: pass - decorator(param1, param2) 返回将用于装饰的实际函数。这里有 3 级嵌套函数。

以上是关于Python 中的装饰器类的主要内容,如果未能解决你的问题,请参考以下文章

用于制作 python 装饰器类的精益接口

在 python memoization 装饰器类中设置 get/set 属性

装饰器、装饰器类与类装饰器(三)

IntelliJ 可以自动创建装饰器类吗?

装饰器类学习小结

python 装饰器