Python函数装饰链路_多装饰器执行解析【教程】

装饰器执行顺序为定义时从下到上、调用时从上到下;如@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。

装饰器链路中的函数包装与调用流程

每个装饰器本质是接收函数、返回新函数的高阶函数。多装饰器组合后,最终函数对象被层层封装:

  • 定义阶段: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 失败