【问题标题】:How do I pass extra arguments to a Python decorator?如何将额外的参数传递给 Python 装饰器?
【发布时间】:2012-04-27 21:46:00
【问题描述】:

我有一个像下面这样的装饰器。

def myDecorator(test_func):
    return callSomeWrapper(test_func)
def callSomeWrapper(test_func):
    return test_func
@myDecorator
def someFunc():
    print 'hello'

我想增强这个装饰器以接受下面的另一个参数

def myDecorator(test_func,logIt):
    if logIt:
        print "Calling Function: " + test_func.__name__
    return callSomeWrapper(test_func)
@myDecorator(False)
def someFunc():
    print 'Hello'

但是这段代码给出了错误,

TypeError: myDecorator() 只接受 2 个参数(给定 1 个)

为什么函数没有自动传递?如何将函数显式传递给装饰器函数?

【问题讨论】:

  • balki:请避免使用 boolean 作为参数,这不是 gd 方法并降低代码的可读性
  • @KitHo -- 这是一个布尔标志,所以使用布尔值是正确的方法。
  • @KitHo -- 什么是“gd”?它“好”吗?

标签: python python-2.7 python-decorators


【解决方案1】:

由于您像调用函数一样调用装饰器,因此它需要返回另一个函数,即实际的装饰器:

def my_decorator(param):
    def actual_decorator(func):
        print("Decorating function {}, with parameter {}".format(func.__name__, param))
        return function_wrapper(func)  # assume we defined a wrapper somewhere
    return actual_decorator

外部函数将获得您显式传递的任何参数,并应返回内部函数。内部函数将传入要装饰的函数,并返回修改后的函数。

通常您希望装饰器通过将其包装在包装函数中来更改函数行为。这是一个在调用函数时可选择添加日志记录的示例:

def log_decorator(log_enabled):
    def actual_decorator(func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            if log_enabled:
                print("Calling Function: " + func.__name__)
            return func(*args, **kwargs)
        return wrapper
    return actual_decorator

functools.wraps 调用将名称和文档字符串等内容复制到包装函数中,使其更类似于原始函数。

示例用法:

>>> @log_decorator(True)
... def f(x):
...     return x+1
...
>>> f(4)
Calling Function: f
5

【讨论】:

  • 并且建议使用functools.wraps -- 它保留被包装函数的原始名称、文档字符串等。
  • @AKX:谢谢,我将其添加到第二个示例中。
  • 所以基本上装饰器总是只接受一个参数,即函数。但是装饰器可以是可能带参数的函数的返回值。这是正确的吗?
  • @balki:是的,没错。令人困惑的是,很多人还会将外部函数(此处为myDecorator)称为装饰器。这对装饰器的用户来说很方便,但在您尝试编写装饰器时可能会感到困惑。
  • 让我困惑的小细节:如果你的log_decorator带默认参数,你不能使用@log_decorator,它必须是@log_decorator()
【解决方案2】:

只是为了提供一个不同的观点:语法

@expr
def func(...): #stuff

等价于

def func(...): #stuff
func = expr(func)

特别是,expr 可以是您喜欢的任何东西,只要它评估为可调用对象。在特别中,expr 可以是一个装饰器工厂:你给它一些参数,它就会给你一个装饰器。因此,了解您的情况的更好方法可能是

dec = decorator_factory(*args)
@dec
def func(...):

然后可以缩短为

@decorator_factory(*args)
def func(...):

当然,因为它看起来decorator_factory 是一个装饰器,人们倾向于命名它来反映这一点。当您尝试遵循间接级别时,这可能会令人困惑。

【讨论】:

  • 谢谢,这真的帮助我理解了发生的事情背后的基本原理。
【解决方案3】:

只是想添加一些有用的技巧,允许装饰器参数可选。它还允许重用装饰器并减少嵌套

import functools

def myDecorator(test_func=None,logIt=None):
    if test_func is None:
        return functools.partial(myDecorator, logIt=logIt)
    @functools.wraps(test_func)
    def f(*args, **kwargs):
        if logIt==1:
            print 'Logging level 1 for {}'.format(test_func.__name__)
        if logIt==2:
            print 'Logging level 2 for {}'.format(test_func.__name__)
        return test_func(*args, **kwargs)
    return f

#new decorator 
myDecorator_2 = myDecorator(logIt=2)

@myDecorator(logIt=2)
def pow2(i):
    return i**2

@myDecorator
def pow3(i):
    return i**3

@myDecorator_2
def pow4(i):
    return i**4

print pow2(2)
print pow3(2)
print pow4(2)

【讨论】:

  • 对不起,对于死灵肿块。如果这种模式被认为是一种好的做法,我很感兴趣?
【解决方案4】:

只是做装饰器的另一种方式。 我发现这种方式最容易绕开我的脑袋。

class NiceDecorator:
    def __init__(self, param_foo='a', param_bar='b'):
        self.param_foo = param_foo
        self.param_bar = param_bar

    def __call__(self, func):
        def my_logic(*args, **kwargs):
            # whatever logic your decorator is supposed to implement goes in here
            print('pre action baz')
            print(self.param_bar)
            # including the call to the decorated function (if you want to do that)
            result = func(*args, **kwargs)
            print('post action beep')
            return result

        return my_logic

# usage example from here on
@NiceDecorator(param_bar='baaar')
def example():
    print('example yay')


example()

【讨论】:

  • 谢谢!花了大约 30 分钟的时间研究了一些令人费解的“解决方案”,这是第一个真正有意义的解决方案。
  • 非常聪明的方式来实现带有可选参数的装饰器,而不需要创建嵌套复杂性。重构我所有的自定义装饰器,谢谢!
【解决方案5】:

现在,如果你想用装饰器 decorator_with_arg 调用函数 function1,在这种情况下,函数和装饰器都接受参数,

def function1(a, b):
    print (a, b)

decorator_with_arg(10)(function1)(1, 2)

【讨论】:

    猜你喜欢
    • 2022-12-05
    • 2018-12-04
    • 2020-04-13
    • 2014-11-17
    • 1970-01-01
    • 2017-12-13
    相关资源
    最近更新 更多