将装饰器功能合并为类

时间:2010-05-02 17:51:27

标签: python class arguments decorator kwargs

需要创建一个可以作为“合并”功能执行所有操作的类。在课堂上,我将改变,处理和添加新论点。

def merge(*arg, **kwarg): # get decorator args & kwargs
    def func(f):
        def tmp(*args, **kwargs): # get function args & kwargs    
            kwargs.update(kwarg) # merge two dictionaries
            return f(*args, **kwargs) # return merged data
        return tmp
    return func

用法:

@other_decorator # return *args and **kwarg
@merge(list=['one','two','three']) # need to merge with @other_decorator
def test(*a, **k): # get merged args and kwargs
    print 'args:', a
    print 'kwargs:', k

1 个答案:

答案 0 :(得分:2)

我不确定我能得到你所要求的。你的实现工作正常,如果你想创建一个任何类型的参数化装饰器,你就不会有两个间接层。

要使合并成为一个类,你可以这样做

class Merge(object):
    def __init__(self, **extra_kws):
        self.extra_kws = extra_kws
    def __call__(self, function):
        def _wrapper(*args, **kws):
            kws.update(self.extra_kws)
            return function(*args, **kws)
        return _wrapper

然后你可以这样做:

@Merge(foo='bar')
def test(*args, **kws):
    print *args
    print **kws

但是你说你想要添加更改并处理新的参数。所以大概你想要装饰器本身就可以了,所以你可以这样做:

test.extra_kws['sun'] = 'dock'

应用装饰器后。在这种情况下,您可能不希望merge成为一个类,但是您希望它生成一个类,以便test被可修改的实例替换:

def merge(**extra_kws):
    class _Merge(object):
        def __init__(self, function):
            self.extra_kws = extra_kws
            self.function = function
        def __call__(self, *args, **kws):
            kws.update(self.extra_kws)
            return self.function(*args, **kws)
    return _Merge

@merge(foo='bar')
def test(*args, **kws):
    print 'args:', args
    print 'kws:', kws

test(sun='dock')
test.extra_kws['trog'] = 'cube'
test(sun='dock')

然后,您可以稍后更改特定装饰功能的关键字。

你也可以使用没有类的函数参数做同样的事情:

def merge(**extra_kws):
    def _decorator(function):
        def _wrapper(*args, **kws):
            kws.update(_wrapper.extra_kws)
            return function(*args, **kws)
        _wrapper.extra_kws = extra_kws
        return _wrapper
    return _decorator

@merge(foo='bar')
def test(*args, **kws):
    print 'kws:', kws

test(sun='dock')
test.extra_kws['trog'] = 'cube'
test(sun='dock')