Python多装饰器执行顺序总结

简介: Python多装饰器是从外到内执行的,再执行被装饰的函数。当然这只是在装饰器中的闭包函数的运行顺序,如果在装饰器函数和闭包函数之前有代码,那运行起来又不一样,具体看下面的例子。

先给出结论:

Python多装饰器是从外到内执行的,再执行被装饰的函数。当然这只是在装饰器中的闭包函数的运行顺序,如果在装饰器函数和闭包函数之前有代码,那运行起来又不一样,具体看下面的例子。


情况1


例如:

# -*- coding:utf-8 -*-
def decorator_a(func):
    def inner_a(*args, **kwargs):
        print 'Get in inner_a'
        return func(*args, **kwargs)
    return inner_a
def decorator_b(func):
    def inner_b(*args, **kwargs):
        print 'Get in inner_b'
        return func(*args, **kwargs)
    return inner_b
@decorator_b
@decorator_a
def f(x):
    print 'Get in f'
    return x * 2
f(1)

上面代码的运行结果为:

Get in inner_b
Get in inner_a
Get in f

由此可见,是先运行的decorator_b,再运行的decorator_a,最后运行的被装饰函数f(x)

这是因为decorator_a装饰器先return 了inner_a, 而decorator_b后面又把inner_a装饰了,最终整个暴露在外面的是inner_b,所以显示inner_b先运行,最终的效果看起来就是装饰器decorator_b先运行。实际上代码在机器上跑的时候是先跑的decorator_a函数,再跑的decorator_b函数


情况2


代码如下:

# -*- coding:utf-8 -*-
def decorator_a(func):
    print 'Get in decorator_a'
    def inner_a(*args, **kwargs):
        print 'Get in inner_a'
        return func(*args, **kwargs)
    return inner_a
def decorator_b(func):
    print 'Get in decorator_b'
    def inner_b(*args, **kwargs):
        print 'Get in inner_b'
        return func(*args, **kwargs)
    return inner_b
@decorator_b
@decorator_a
def f(x):
    print 'Get in f'
    return x * 2
f(1)

上面代码运行的结果为:

Get in decorator_a
Get in decorator_b
Get in inner_b
Get in inner_a
Get in f

现在的结果和情况1得出的运行顺序结论不一致了,在每个装饰器中,装饰器函数和内层的闭包函数之间的代码是先运行的decorator_a 再运行的decorator_b。这和情况1的结论恰好相反。代码位置不同,运行的顺序也不。同。

再看一种情况


情况3


# -*- coding:utf-8 -*-
def decorator_a(func):
    print 'Get in decorator_a'
    def inner_a(*args, **kwargs):
        print 'Get in inner_a'
        return func(*args, **kwargs)
    return inner_a
def decorator_b(func):
    print 'Get in decorator_b'
    def inner_b(*args, **kwargs):
        print 'Get in inner_b'
        return func(*args, **kwargs)
    return inner_b
@decorator_b
@decorator_a
def f(x):
    print 'Get in f'
    return x * 2

与情况2相比,这里没有对函数f进行调用,去掉了情况2中的f(1)这一行代码。只是声明了被装饰器装饰的函数f(x),但是没有进行调用。但是运行上面的代码是有结果输出的。结果如下:

Get in decorator_a
Get in decorator_b

可看到只运行了装饰器函数和内层闭包函数之间的代码。为什么会这样呢。那是因为在装饰器中的内层闭包函数被return了,而装饰器也相当于是函数调用,只是闭包的函数需要在最后被return出来,在调用被装饰函数f(x)时,装饰其中return的inner_a和inner_b才会被执行。return一个函数的名字,这个函数是没有被执行的,函数名带有括号和参数才会去执行,没有带括号的函数名只是一个对象而已,没有被执行。

因此上面inner函数和装饰器函数之间的代码会执行,及时不调用被装饰的f(x)函数


小tips


def decorator_a(func):
    print 'Get in decorator_a'
    def inner_a(*args, **kwargs):
        print 'Get in inner_a'
        print "in a, args ", args
        print "in a, kwargs ", kwargs
        kwargs.update({"params": "1234"})
        return func(*args, **kwargs)
    return inner_a
def decorator_b(func):
    print 'Get in decorator_b'
    def inner_b(*args, **kwargs):
        print 'Get in inner_b'
        print "in b, args ", args
        print "in b, kwargs ", kwargs
        return func(*args, **kwargs)
    return inner_b
@decorator_b
@decorator_a
def f(x, params):
    print 'Get in f'
    print "params: ", params
    return x * 2
f(*(1, ))

上面代码运行的结果如下:

Get in decorator_a
Get in decorator_b
Get in inner_b
in b, args  (1,)
in b, kwargs  {}
Get in inner_a
in a, args  (1,)
in a, kwargs  {}
Get in f
params:  1234

上面运用的小tips是在装饰器中给被装饰的函数加参数,定义函数f时,他有两个参数,但是在调用他时。f(*(1, ))是只传了一个参数。

在decorator_b中给kwargs字典加了一个key为params的字典,这样就能在函数f中使用这个params,虽然在调用f的地方没有传这个参数。

这个tips可以用在flask中@route装饰器的下面,能用一个装饰器对flask请求的参数进行合法验证检查。

例如:

@app.route("/index/<name>/test")
@parse_params
def test(name, params):
    return "hello world"

test函数有两个参数:name和params。其中name这个是url中的变量,flask会帮助处理。对于params参数,可以在装饰器parse_params中对request.args做一些校验后生成一个params字典加在装饰器parse_params的闭包函数参数的kwargs中,类似于上面的那个例子。

这个用在flask中能用完全在于,flask中调用我们注册的def test函数是这样处理的:

self.view_functions[rule.endpoint](**req.view_args)

 

上面传的参数就是**req.view_args的形式传的。这种是不定参数的传法

目录
相关文章
|
16天前
|
数据安全/隐私保护 开发者 Python
深入浅出Python装饰器
【8月更文挑战第4天】装饰器在Python中是一个既强大又神秘的功能,它允许开发者在不修改原有函数代码的情况下增加额外的功能。本文旨在通过浅显易懂的语言和实例,带领读者一步步揭开装饰器的神秘面纱,理解其背后的原理,并学会如何在实际开发中应用它们。
|
19天前
|
测试技术 开发者 Python
Python 编程中的装饰器深入解析
【8月更文挑战第1天】本文将通过实例和代码演示,深入探讨 Python 中装饰器的概念、用法和高级应用。我们将从基础开始,逐步过渡到如何自定义装饰器,并展示其在日志记录、性能测试等场景下的实际用途。文章最后还将讨论装饰器的常见误区和最佳实践。
Python 装饰器“高级”使用
本文聚焦两个有意思的点 1. 无参和有参装饰器。 @deco vs @deco(arg1,arg2)。 2. 多层装饰器场景。
|
19天前
|
缓存 Python
代码之美:探索Python中的装饰器
【7月更文挑战第31天】在编程的世界里,装饰器就像是一位神奇的艺术家,它能够为我们的代码添加额外的功能,而不改变原有代码的结构。本文将带领你走进Python的装饰器世界,通过实例学习如何创造和使用装饰器,让你的代码更加优雅和高效。
37 18
|
16天前
|
Python
探索Python中的装饰器:从入门到实践
【8月更文挑战第4天】在Python的世界中,装饰器是一把双刃剑,它既能美化代码,又能提升效率。本文将带你一探究竟,通过实例学习如何定义、使用以及深入理解装饰器背后的原理。我们将一起揭开这层神秘的面纱,让装饰器成为你编程工具箱中的又一利器。
31 9
|
16天前
|
测试技术 开发者 Python
揭秘Python中的装饰器:从入门到精通
【8月更文挑战第4天】装饰器,在Python中是一块神奇的“画布”,它允许开发者在不修改原有函数代码的情况下增加额外的功能。本文将通过实际的代码示例,带你一探究竟,从基础使用到高级技巧,逐步揭开装饰器的神秘面纱。
|
13天前
|
测试技术 开发者 Python
翻天覆地!Python装饰器,如何让代码起死回生?
【8月更文挑战第6天】在软件开发领域,提高代码的质量始终是核心目标之一。Python作为一种功能丰富的高级语言,提供了多种手段来实现这一目标,装饰器便是其中之一。本文通过问答形式,深入解析了装饰器的概念、基本语法及其实现机制。装饰器允许在不改变原函数的基础上添加新功能,其基本语法为使用`@`符号后跟装饰器函数名。此外,还探讨了装饰器如何通过增强代码的模块性和灵活性来提升整体质量,并举例说明了装饰器在类方法中的应用。总之,装饰器是一种强大的工具,可以帮助开发者以更简洁、模块化的方式扩展功能,同时保持代码的整洁和可维护性。
25 3
|
17天前
|
设计模式 开发者 Python
探索Python中的装饰器:从基础到高级应用
【8月更文挑战第3天】本文将深入探讨Python编程中一个强大而灵活的特性——装饰器。我们将从理解装饰器的基本概念开始,通过实际代码示例,逐步展示如何创建和使用它们。文章旨在揭示装饰器背后的魔法,帮助开发者解锁其在代码重用和扩展性方面的潜力。
|
19天前
|
Python
探索Python中的装饰器:从入门到实践
【8月更文挑战第1天】本文将带领读者深入理解Python装饰器的概念,并透过代码示例展示如何利用装饰器来增强函数功能。我们将一起探索装饰器的工作原理,学习如何自定义装饰器,并探讨其在实战中的应用。通过阅读本篇文章,你将能够掌握装饰器的使用,为你的Python项目增添强大的功能。
|
19天前
|
测试技术 Python
Python中的装饰器:简化你的代码
【8月更文挑战第1天】在编程的世界中,我们常常追求的是简洁与高效。Python作为一种高级编程语言,其独特的特性之一就是“装饰器”。装饰器不仅能够让我们以简洁的方式修改函数的行为,还能在不改变原函数定义的前提下增加额外的功能。本文将通过实例介绍如何在Python中使用装饰器,以及如何自定义装饰器来解决实际问题。让我们一起探索装饰器的奥秘,让你的代码更加优雅。