装饰装饰:试着理解它

ash*_*ods 21 python decorator

我正在尝试了解装饰装饰器,并想尝试以下内容:

假设我有两个装饰器并将它们应用于函数hello:

def wrap(f):
    def wrapper():
        return " ".join(f())
    return wrapper


def upper(f):
    def uppercase(*args, **kargs):
        a,b = f(*args, **kargs)
        return a.upper(), b.upper()
    return uppercase

@wrap
@upper
def hello():
    return "hello","world"

print hello()
Run Code Online (Sandbox Code Playgroud)

然后我必须开始为其他函数添加其他装饰器,但一般来说,wrap装饰器将"包装所有这些"

def lower(f):
    def lowercase(*args, **kargs):
        a,b = f(*args, **kargs)
        return a.lower(), b.lower()
    return lowercase

@wrap
@lower
def byebye():
    return "bye", "bye"
Run Code Online (Sandbox Code Playgroud)

现在我如何写一个装饰器,我可以装饰我的下装饰器和上装饰器:

@wrap
def lower():
    ...

@wrap
def upper():
    ...
Run Code Online (Sandbox Code Playgroud)

通过以下方式实现与上述相同的结果:

@upper
def hello():
    ...

@lower
def byebye():
    ...
Run Code Online (Sandbox Code Playgroud)

ken*_*ytm 24

def upper(f):
    @wrap
    def uppercase(*args, **kargs):
        a,b = f(*args, **kargs)
        return a.upper(), b.upper()
    return uppercase
Run Code Online (Sandbox Code Playgroud)

Python中的装饰器

 @foo
 def bar(...): ...
Run Code Online (Sandbox Code Playgroud)

等于

 def bar(...): ...
 bar = foo(bar)
Run Code Online (Sandbox Code Playgroud)

你想得到的效果

@wrap
@upper
def hello(): ....
Run Code Online (Sandbox Code Playgroud)

hello = wrap(upper(hello))
Run Code Online (Sandbox Code Playgroud)

所以wrap应该在返回值上调用upper:

def upper_with_wrap(f):
   def uppercase(...): ...
   return wrap(uppercase)
Run Code Online (Sandbox Code Playgroud)

这也相当于在该函数上应用装饰器:

def upper_with_wrap(f):
   @wrap
   def uppercase(...): ...
   # ^ equivalent to 'uppercase = wrap(uppercase)'
   return uppercase
Run Code Online (Sandbox Code Playgroud)


Boa*_*niv 9

这是一个通用(并且有点复杂)的解决方案,用于装饰装饰器(Yay!).

# A second-order decorator
def decdec(inner_dec):
    def ddmain(outer_dec):
        def decwrapper(f):
            wrapped = inner_dec(outer_dec(f))
            def fwrapper(*args, **kwargs):
               return wrapped(*args, **kwargs)
            return fwrapper
        return decwrapper
    return ddmain

def wrap(f):
    def wrapper():
        return " ".join(f())
    return wrapper


# Decorate upper (a decorator) with wrap (another decorator)
@decdec(wrap)
def upper(f):
    def uppercase(*args, **kargs):
        a,b = f(*args, **kargs)
        return a.upper(), b.upper()
    return uppercase

@upper
def hello():
    return "hello","world"

print hello()
Run Code Online (Sandbox Code Playgroud)