
本文详解python装饰器内部函数(inner)为何能访问调用时传入的参数num,揭示闭包机制与函数调用链的本质关系,并通过代码执行流程图解说明装饰器语法糖的实际展开过程。
在Python中,inner函数之所以能访问num,根本原因并非它“提前知道”参数名,而是因为num是inner被调用时动态传入的实参——它属于inner自身的局部作用域,而非来自外层facto_decorator的参数列表。
我们来逐步拆解装饰器的执行逻辑:
1. 装饰器语法糖的等价展开
@facto_decorator 并非魔法,它只是语法糖。以下两种写法完全等价:
@facto_decorator
def facto(num):
...等价于:
立即学习“Python免费学习笔记(深入)”;
def facto(num):
...
facto = facto_decorator(facto) # ← 关键:重绑定facto为inner函数!此时,facto 已不再是原始递归函数,而是 facto_decorator 返回的 inner 函数对象。
2. 调用时的真正执行路径
当你执行 facto(a)(例如 a = 5)时,实际发生的是:
facto(a) # → 实际调用的是 inner(5) # 展开为: inner(5) # → num = 5 进入 inner 的局部作用域
因此,num 是 inner 的形参(parameter),每次调用 inner(...) 都会为其创建新的局部命名空间,num 自然可被直接读写。
⚠️ 注意:facto_decorator 函数本身确实不接收 num —— 它只接收被装饰的函数 func(即原始 facto)。num 是在后续调用阶段才出现的,与 facto_decorator 的执行时刻无关。
3. 闭包的作用:捕获 func 和 memory,而非 num
inner 是一个闭包函数,它封闭(enclose)了定义时所在作用域的变量:
- ✅ func(原始 facto 函数对象)
- ✅ memory(外部字典,用于缓存)
但 num 不属于闭包变量——它是运行时传入的局部参数。闭包保存的是 静态引用,而 num 是 动态值,二者作用域层级不同。
你可以用 inner.__closure__ 验证:
print(facto.__closure__) # 输出: (| , | ) # 对应 func 和 memory;没有 num 的 cell! |
总结:关键认知澄清
- inner(num) 的 num 是函数签名的一部分,和普通函数参数无异;
- 装饰后 facto = inner,所以 facto(5) 就是 inner(5);
- facto_decorator 只负责“制造并返回 inner”,不参与后续调用;
- 缓存逻辑成立,是因为 inner 在每次调用中都能拿到当前 num 去查 memory 字典。
理解这一点,就掌握了装饰器最核心的“函数替换 + 闭包增强”范式。










