python - 使用 Python 装饰器为方法添加方法

标签 python oop decorator python-decorators theano

我希望能够根据某种标准格式调用方法:

outputs = obj.meth(in_0, in_1, ...)

,其中输出是一个数组元组,每个输入是一个数组。

然而,在大多数情况下,我只返回一个数组,不想为了标准格式而被迫返回长度为 1 的元组。 (我的实际格式问题更复杂,但现在我们坚持这个解释。)

我希望能够定义一个类:

class _SomeClass(object):

    def __init__(self):
        self._amount_to_add = 1

    @single_return_format
    def add_one(self, x):
        return x+self._amount_to_add

然后可以这样调用它:

obj = _SomeClass()
assert obj.add_one(3) == 4
assert obj.add_one.standard_format(3)==(4, )

问题是:如何定义装饰器以允许这种行为?

我试过:

def single_return_format(fcn):
    fcn.standard_format = lambda *args: (fcn(*args), )
    return fcn

,但它在第二个断言行上失败了:

TypeError: add_one() takes exactly 2 arguments (1 given)

因为 add_one 需要“self”作为参数,而装饰器修改函数时对象甚至还没有创建。

Stack,我该怎么做?


注意事项:

1) 我知道我可以使用基类和继承来做到这一点,但是当类中有多个方法要以这种方式装饰时,这就会成为一个问题。

2) 实际问题来自使用 theano - 标准格式是 outputs, updates = fcn(*inputs),但大多数函数不返回任何更新,所以你希望能够以自然的方式定义这些函数,但仍然可以选择根据此标准接口(interface)调用它们。

最佳答案

这确实是一个问题,因为从函数中检索“绑定(bind)”方法的方式并没有考虑这种方式。

我看到两种方式:

  1. 您可以只包装函数:

    def single_return_format(fcn):
        # TODO Do some functools.wraps here...
        return lambda *args, **kwargs: (fcn(*args, **kwargs), )
    

    不要乱用 .standard_format,而只是函数的替换。所以函数可以定义自己为返回一个值,但只能作为返回元组被调用。

  2. 如果这不是您想要的,您可以定义一个类来装饰方法,它会覆盖 __get__ 并以“实时方式”进行包装。当然,它也可以重新定义 __call__,以便它也可用于(独立的、非方法的)函数。

关于python - 使用 Python 装饰器为方法添加方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26725593/

相关文章:

python - Raspberry Pi 2 - 无法启用/dev/i2c-0

python - 如何递归翻译DNA? (没有 while/for 循环,也没有翻译函数)在 Python 中

python - 为什么 key = model_instance.key 没有!!!在 db 到 ndb 文档的旁边?

python - 我如何编写装饰器来将某些内容包装在带参数的上下文管理器中?

python 3 : how do I run a function over a list of lists of possible parameters and returning a dictionary of all results

python - 仅当单独列中的差异在 [-n, +n] 范围内时,才在公共(public)列上加入两个 DataFrame

objective-c - 如何共享其他类的方法

perl - 为什么将 Perl OO 简单地用作数据封装技术?

c++ - 多重继承。这个实现 `+=` 的继承模型背后的想法是什么?

.net - 在 SimpleInjector 中有条件地注册基于泛型类型约束的命令装饰器