1

「マージ」機能としてすべてのことを行うクラスを作成する必要があります。クラス i では、新しい引数を変更、処理、および追加します。

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
4

1 に答える 1

2

あなたが求めていることを完全に理解しているかどうかはわかりません。実装は問題なく動作し、任意の種類のパラメーター化されたデコレーターを作成したい場合、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'

デコレータが適用された後。その場合、おそらくマージをクラスにしたくないでしょうが、クラスを生成したいので、それ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')
于 2010-05-02T21:07:36.987 に答える