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 装饰器执行顺序问题详解
第 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
关键点:
- 应用顺序:
@decorator_b先应用到my_function,然后@decorator_a应用到前一步的结果上。相当于decorator_a(decorator_b(my_function))。 - 执行顺序:调用时,最外层的
decorator_a的wrapper先执行,然后才是内层的decorator_b的wrapper,最后是原函数。
带参数的装饰器也一样,只是多了一层嵌套。记住这个嵌套模型: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 在最里面


