【问题标题】:Class level decorator类级装饰器
【发布时间】:2012-08-07 15:16:53
【问题描述】:

我正在尝试使用以下代码进行一种类型处理函数注册:

types = {}
def type_handler(name):
    def wrapper(f):
        types[name] = f
        return f
    return wrapper

@type_handler('a')
def handle_a(a):
    ...

@type_handler('b'):
def handle_b(b):
    ...

def handle(x):
    types[x.name](x)

这很好用,但现在我希望它在一个类中工作。
我试过这个:

class MyClass(object):

    types = {}
    def type_handler(name):
        def wrapper(f):
            types[name] = f ## global name 'types' is undefined
            return f
        return wrapper

    @type_handler('a')
    def handle_a(self, a):
        ...

    @type_handler('b'):
    def handle_b(self, b):
        ...

    def handle(self, x):
        self.types[x.name](self, x)

但上面写着global name 'types' is undefined
我尝试将其更改为

    def type_handler(name):
        def wrapper(f):
            MyClass.types[name] = f ## global name 'MyClass' is undefined
            return f
        return wrapper

但现在是global name 'MyClass' is undefined
我能做些什么来完成这项工作?

我知道我可以这样做:

def handle(self, x):
    self.__getattribute__('handle_%s' % x.name)(self, x)

但我更喜欢函数注册而不是基于名称的查找。

【问题讨论】:

标签: python class decorator


【解决方案1】:

我同意 Sven Marnach 的观点。你应该使用“更少的魔法”。不过,这里有一个快速解决方法:

#decorator is declared outside of class and type dict is passed in as an argument
def type_handler(name, type_dict):
    def wrapper(f):
        type_dict[name] = f
        return f
    return wrapper

class MyClass(object):

    types = {}

    @type_handler('a', types)
    def foo_a(self, a):
        pass

    @type_handler('b', types)
    def foo_b(self, b):
        pass

第 2 版

这个使用一个类作为装饰器:

class TypeHandler(object):
    def __init__(self, type_dict):
        self.types = type_dict

    def __call__(self, name):
        def wrapper(f):
            self.types[name] = f
            return f
        return wrapper


class MyClass(object):
    types = {}
    thandle = TypeHandler(types)

    @thandle('a')
    def foo_a(self, a):
        pass

【讨论】:

  • 这在装饰器调用上有点多余。
猜你喜欢
  • 2019-06-28
  • 1970-01-01
  • 2017-09-25
  • 2012-08-14
  • 1970-01-01
  • 2014-01-14
  • 2020-01-11
  • 2015-06-24
  • 1970-01-01
相关资源
最近更新 更多