Объединить функцию декоратора как класс

#python #class #arguments #decorator #keyword-argument

#python #класс #аргументы #декоратор #ключевое слово-аргумент

Вопрос:

Необходимо создать класс, который будет выполнять все действия как функция «merge». В классе я буду изменять, обрабатывать и добавлять новые аргументы.

 def merge(*arg, **kwarg): # get decorator args amp; kwargs
    def func(f):
        def tmp(*args, **kwargs): # get function args amp; 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. Что вы хотите, чтобы декоратор класса делал? В примере показан только декоратор функции.

2. Необходимо создать класс, который будет выполнять все действия как функция «merge». В классе я буду изменять, обрабатывать и добавлять новые аргументы. Функция «слияние» работает, но у меня слишком много подфункций процесса.

Ответ №1:

Я не уверен, что вполне понимаю, о чем вы спрашиваете. Ваша реализация работает нормально, и вам не обойтись без двух уровней косвенности, если вы хотите создать параметризованный декоратор любого вида.

Чтобы сделать слияние классом, вы могли бы сделать это

 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')