Python装饰器(3)多装饰器装饰

内容纯属个人理解,不对之处,欢迎指正。

多装饰器示例

def deco1(func):
    '''i am deco1'''

    print('---deco1---')

    def wrapper1(*args, **kwargs):
        '''i am wrapper1'''

        print('---wrapper1:before func---')
        func(*args, **kwargs)
        print('---wrapper1:after func---')
    return wrapper1


def deco2(func):
    '''i am deco2'''

    print('---deco2---')

    def wrapper2(*args, **kwargs):
        '''i am wrapper2'''

        print('---wrapper2:before func---')
        func(*args, **kwargs)
        print('---wrapper2:after func---')
    return wrapper2


@deco1
@deco2
def foo():
    '''i am foo'''
    print('---foo---')


if __name__ == '__main__':
    print('---start---')
    foo()

运行结果:

---deco2---
---deco1---
---start---
---wrapper1:before func---
---wrapper2:before func---
---foo---
---wrapper2:after func---
---wrapper1:after func---

通过现象看本质。我们注意到,程序开始执行后,装饰函数是最先开始执行的,然后再执行的foo。原因就是Python解释器在执行到foo定义处时,检测到有两个装饰器,所以立即执行其中的装饰代码对函数进行装饰,此时foo函数便具有了额外的功能。
接下来在执行foo时可以看到:多装饰器的装饰顺序是由内到外,而执行顺序是由外到内

多装饰过程拆分

fn2 = deco2(foo)  # ---deco2---
fn1 = deco1(fn2)  # ---deco1---
foo = fn1 
print(foo)  # .wrapper1 at 0x7f18baf4e620>

foo函数作为参数传递到deco2,deco2执行;然后将结果作为参数传递到deco1,deco1执行;然后将结果赋值给foo,此时的foo即为装饰后的函数wrapper1,如果想保留原函数相关信息,可以使用functools中的wraps函数。

你可能感兴趣的:(Python装饰器(3)多装饰器装饰)