装饰器执行顺序为定义时从下到上、调用时从上到下;如@dec1@dec2修饰myfunc,等价于myfunc = dec1(dec2(myfunc)),dec2先包装原函数,dec1再包装dec2结果,调用时先执行dec1返回的wrapper,再触发dec2的wrapper,最终执行原逻辑。

装饰器执行顺序:从下到上,调用时从上到下
多个装饰器叠加时,Python会按定义顺序从下往上应用(即离函数最近的先执行),但实际调用时,外层装饰器包裹内层,形成“套娃”结构。比如:
@dec1
@dec2
def myfunc():
pass
等价于 myfunc = dec1(dec2(myfunc))。dec2 先被调用并返回一个新函数,再作为参数传给 dec1。
装饰器链路中的函数包装与调用流程
每个装饰器本质是接收函数、返回新函数的高阶函数。多装饰器组合后,最终函数对象被层层封装:
立即学习“Python免费学习笔记(深入)”;
- 定义阶段:dec2 包装原始函数 → dec1 包装 dec2 的返回结果
- 调用阶段:执行最外层(dec1 返回的函数)→ 内部触发 dec2 返回的函数 → 最终执行原始逻辑
- 中间装饰器可控制是否调用 inner 函数(即是否放行执行链)
带参数装饰器参与链路时的注意事项
如果某个装饰器本身带参数(如 @retry(max_tries=3)),它必须是三层嵌套结构:
def retry(max_tries):
def decorator(func):
def wrapper(*args, **kwargs):
# 实际逻辑
return func(*args, **kwargs)
return wrapper
return decorator
在装饰链中,它仍遵循“从下到上构建、从上到下调用”规则,只是多了一层参数解析过程。
调试多装饰器链路的小技巧
想看清执行路径,可在每个 wrapper 中加 print 或 logging:
- 在 wrapper 开头打印 “进入 [装饰器名]”
- 在 wrapper 结尾(或调用 inner 前)打印 “即将进入下一层”
- 配合函数名和 id() 可观察对象是否被真正替换
- 使用 functools.wraps 保持原函数元信息,避免调试时 getsource 失败










