Python 装饰器执行顺序问题详解

我用 flask 框架做一个网站的过程中,要用到装饰器,缩减版代码如下:

def testwra(func): print "testwra" def _wrapper(): print "wra" r = func() return r print "end" return _wrapper

[@testwra](/user/testwra) [@app](/user/app).route('/users', methods=['GET']) def get_users(): return ['33', '44']

这里,正常的执行顺序不应该是 testwra->先后打印 testwra 和 end->调用_wrapper->打印 wra->调用 func()函数->返回 list->返回 r,但是我这里在浏览器中输入 url ( http://localhost:5000/users ),却不会打印 wra,好像是没有进入_wrapper 函数中。 而且还会报错 TypeError: 'list' object is not callable list 不是可以作为返回值的吗?


Python 装饰器执行顺序问题详解

10 回复

第 0 个问题,请注意排版好吧
第 1 个问题,app.route 会把它下面包裹的函数作为请求处理函数,这时候还没被 testwra 装饰自然没有打印 wra
第 2 个问题,请求处理函数只能返回(字符串,response,tuple )其中一种,你返回 list 不合法,请去看 flask/app.py/make_response 源码


装饰器的执行顺序是“由内向外”应用,但“由外向内”执行。简单说,离函数定义最近的装饰器最先被应用(包装在最内层),但最后才执行其包装逻辑。

看个例子就明白了:

def decorator_a(func):
    print("Applying decorator_a")
    def wrapper():
        print("Executing decorator_a's wrapper")
        func()
    return wrapper

def decorator_b(func):
    print("Applying decorator_b")
    def wrapper():
        print("Executing decorator_b's wrapper")
        func()
    return wrapper

@decorator_a
@decorator_b
def my_function():
    print("Inside my_function")

print("--- Calling function ---")
my_function()

输出:

Applying decorator_b
Applying decorator_a
--- Calling function ---
Executing decorator_a's wrapper
Executing decorator_b's wrapper
Inside my_function

关键点:

  1. 应用顺序@decorator_b先应用到my_function,然后@decorator_a应用到前一步的结果上。相当于decorator_a(decorator_b(my_function))
  2. 执行顺序:调用时,最外层的decorator_awrapper先执行,然后才是内层的decorator_bwrapper,最后是原函数。

带参数的装饰器也一样,只是多了一层嵌套。记住这个嵌套模型:decorator_a(decorator_b(func))

一句话总结:装饰器从下往上堆叠,但执行时从上往下剥开。

更正:请求处理函数只能返回(字符串,response,tuple , WSGI callable )其中一种

装饰器不是从上向下依次执行的吗,此时 testwra 不是已经先装饰了吗,为什么说没被装饰呢

我是电脑端排好版发的,发完再看排版就乱了,不知道这次什么情况

最重要的事情说三遍,

装饰器是在编译时就执行,而不是调用时;装饰器只对函数进行装饰,不对装饰器进行装饰,谁贴函数近谁先。
装饰器是在编译时就执行,而不是调用时;装饰器只对函数进行装饰,不对装饰器进行装饰,谁贴函数近谁先。
装饰器是在编译时就执行,而不是调用时;装饰器只对函数进行装饰,不对装饰器进行装饰,谁贴函数近谁先。

这是关键。

<br><br>def func():<br> pass<br>
func = deco(func) 的语法糖。这样能明白?

所以
<br><br><br>def func():<br> pass<br>
func = deco2(func); func = deco1(func)
也是 func = deco1(deco2(func))

这样明白?

明白了,感谢🙏🏻

了解了,感谢🙏🏻🙏🏻

不要理解成从上到下,理解成从外到内就好了,func 在最里面

回到顶部