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