如何将额外的参数传递给Python装饰器?

bal*_*lki 79 python python-2.7 python-decorators

我有一个像下面的装饰.

def myDecorator(test_func):
    return callSomeWrapper(test_func)
def callSomeWrapper(test_func):
    return test_func
@myDecorator
def someFunc():
    print 'hello'
Run Code Online (Sandbox Code Playgroud)

我想增强这个装饰器来接受另一个如下所示的参数

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

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

TypeError:myDecorator()只需要2个参数(给定1个)

为什么函数不会自动通过?如何将函数显式传递给装饰器函数?

int*_*jay 141

因为你像一个函数一样调用装饰器,它需要返回另一个函数,它是实际的装饰器:

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
Run Code Online (Sandbox Code Playgroud)

外部函数将被赋予您明确传递的任何参数,并应返回内部函数.内部函数将传递函数进行修饰,并返回修改后的函数.

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

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
Run Code Online (Sandbox Code Playgroud)

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

用法示例:

>>> @log_decorator(True)
... def f(x):
...     return x+1
...
>>> f(4)
Calling Function: f
5
Run Code Online (Sandbox Code Playgroud)

  • 建议使用[`functools.wraps`](http://docs.python.org/library/functools.html#functools.wraps) - 它保留包装函数的原始名称,docstring等. (10认同)
  • 让我困惑的小细节:如果你的“log_decorator”采用默认参数,则不能使用“@log_decorator”,它必须是“@log_decorator()” (3认同)
  • 所以基本上装饰器总是只接受一个参数,即函数。但是装饰器可以是一个可能带参数的函数的返回值。这样对吗? (2认同)
  • @balki:是的,这是对的.令人困惑的是,许多人也会将外部函数(这里的`myDecorator`)称为装饰器.这对于装饰者的用户来说很方便,但是当你试图写一个时,可能会让人感到困惑. (2认同)

Kat*_*iel 43

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

@expr
def func(...): #stuff
Run Code Online (Sandbox Code Playgroud)

相当于

def func(...): #stuff
func = expr(func)
Run Code Online (Sandbox Code Playgroud)

特别是,expr只要它评估为可调用,就可以是你喜欢的任何东西.在特别具体的,expr可以是装饰厂:你给它的一些参数,它给你一个装饰.因此,了解您的情况可能更好

dec = decorator_factory(*args)
@dec
def func(...):
Run Code Online (Sandbox Code Playgroud)

然后可以缩短为

@decorator_factory(*args)
def func(...):
Run Code Online (Sandbox Code Playgroud)

当然,因为它看起来decorator_factory装饰者,所以人们倾向于将其命名为反映.当您尝试遵循间接级别时,这可能会令人困惑.

  • 谢谢,这确实帮助我理解了正在发生的事情背后的基本原理。 (3认同)

Ale*_*nov 19

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

import functools

def myDecorator(test_func=None,logIt=None):
    if not test_func:
        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)
Run Code Online (Sandbox Code Playgroud)


Rob*_*Fey 7

只是做装饰的另一种方法。我发现这种方式最容易缠住我的头。

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()
Run Code Online (Sandbox Code Playgroud)

  • 谢谢你!我花了大约 30 分钟研究一些令人费解的“解决方案”,这是第一个真正有意义的解决方案。 (5认同)
  • 使用可选参数实现装饰器的非常聪明的方法,而不需要创建嵌套复杂性。将我所有的自定义装饰器重构为此,谢谢! (2认同)