Python Decorator无法访问被调用的方法

时间:2018-06-08 02:01:55

标签: python python-3.x class python-decorators attributeerror

我正在尝试为一个将调用第二个方法的方法编写装饰器。当我运行代码时,我收到错误:
AttributeError: 'Backoff' object has no attribute 'formatter'

简化,代码为:

class Backoff:
    def __init__(self, f):
        self.f = f

    def __call__(self, *args, **kwargs):
        n = 1
        while n < 11:
            try:
                return self.f(self, *args, **kwargs)
            except FooError as e:
                <handle error>
                time.sleep((2 ** n) + (random.randint(0, 1000) / 1000))
                n = n + 1

class SomeClass:
    def __init__(self):
        pass
    @Backoff
    def first_method(self, foo, bar):
        return self.formatter(foo, bar)

    def formatter(self, x, y):
        return some_function_to_format(x, y)

如何以装饰者可识别的方式将第二种方法传递给第一种方法?

任何帮助都会很棒!

1 个答案:

答案 0 :(得分:2)

您正在将方法传递给Backoff,将其指定为实例变量,然后调用它。这不会绑定到实例,直到它在Backoff.__call__中被调用,然后绑定到Backoff的实例,该实例没有formatter属性。

根据您的实例方法需要访问的内容,可能有一个直接的解决方案(即,如果它只需要引用类或静态方法,它可以直接调用完全限定名称)。但是,如果您需要实例方法来引用实例属性,我建议不要使用类装饰器。使用函数装饰器不会遇到这些问题,您可以创建一个闭包并返回一个具有相同调用签名的函数。