如何将未知的参数列表发送给python装饰器?

时间:2017-06-06 18:21:27

标签: python python-decorators

我创建了一个python装饰器,如下所示。我希望这个装饰器接受一个未知的参数列表。但是下面的代码不起作用。

#!/usr/bin/env python

from functools import wraps

def my_decorator(decorated_function, **kwargs):

    print "kwargs = {}".format(kwargs)
    @wraps(decorated_function)
    def inner_function(*args, **kwargs):
        print "Hello world"
        return decorated_function(*args, **kwargs)
    return inner_function

@my_decorator(arg_1="Yolo", arg2="Bolo")
def my_func():
     print "Woo Hoo!"


my_func()

当我运行它时,我收到此错误:

 File "./decorator_test.py", line 14, in <module>
     @my_decorator(arg_1="Yolo", arg2="Bolo")
 TypeError: my_decorator() takes exactly 1 argument (0 given)

为什么**kwargs不接受我发送到装饰器的多个参数?我该如何解决?

1 个答案:

答案 0 :(得分:2)

基本上,当你有一个接受参数的装饰器时,你可以调用它两次,一次使用关键字参数,然后再使用要装饰的函数。

@deco语法之前,只需将函数传递给装饰器函数即可修饰函数

func = deco(func)

如果你想包含其他关键字参数,你可以同时传入它们

func = deco(func, arg=True, arg2=5)

但是@deco语法不允许这样做,它更像是这样,因此您可以通过装饰器函数调用函数返回

func = deco(arg=True, arg2=5)(func)

要使用@deco语法执行此操作,您将创建一个装饰器,用于测试是仅使用函数(装饰部分)调用它,还是使用关键字参数调用它(设置为要传递函数的装饰器)。

def deco_with_kwargs(func=None, **kwargs):

    def deco(func_):

        def wrapped_func(*fargs, **fkwargs):
            print kwargs
            return func_(*fargs, **fkwargs)
        return wrapped_func

    if func:
        return deco(func)
    else:
        return deco

您可以在没有任何关键字参数的情况下使用它。

@deco_with_args
def my_func():
    return 1

或者您可以使用关键字参数调用它。在这种情况下,您实际上正在调用装饰器函数,该函数返回另一个装饰器函数,然后调用该函数来实际装饰函数。

@deco_with_args(test=True)
def my_func():
    return 1