python闭包的核心是内部函数绑定外层变量名而非值,需满足嵌套结构、引用自由变量、外部函数返回内部函数对象三条件;变量查找延迟至调用时,故循环中易出现所有闭包共享最终值的问题。

Python闭包的核心在于:内部函数引用了外部函数的局部变量,且外部函数返回该内部函数(而非调用它),此时这个内部函数就构成了闭包。关键不是“捕获值”,而是“绑定变量名到外层作用域的查找链”,而真正的值是在闭包被调用时才取的。
闭包形成的三个必要条件
一个函数要成为闭包,必须同时满足:
- 存在嵌套函数结构(内部函数定义在外部函数内部)
- 内部函数引用了外部函数的非全局、非参数的局部变量(即自由变量,free variable)
- 外部函数返回内部函数对象(不是调用结果,例如 return inner,而非 return inner())
变量绑定发生在定义时,但值读取发生在调用时
这是最容易误解的一点:闭包绑定的是变量名和其所在的外层作用域,而不是某个时刻的值。也就是说,Python 不会把变量当时的值“快照”下来,而是记下“去哪找这个变量”。如果外部函数执行完后,那个变量还存在(比如被闭包引用),它就不会被销毁,而是保留在闭包的 __closure__ 中。
典型反例是循环中创建多个闭包:
立即学习“Python免费学习笔记(深入)”;
funcs = []
for i in range(3):
funcs.append(lambda: i)
print([f() for f in funcs]) # 输出 [2, 2, 2],不是 [0, 1, 2]原因:所有 lambda 都绑定到同一个变量 i,而循环结束时 i == 2;调用时统一去当前作用域找 i,得到的都是最终值。
修复方式是让每次迭代绑定当前值,例如:
- 用默认参数固化: lambda x=i: x
- 用额外闭包封装: make_func = lambda x: lambda: x; funcs.append(make_func(i))
闭包对象如何保存外部变量
当闭包形成后,可通过 func.__closure__ 查看它引用的自由变量元组,每个元素是 cell 对象,其 cell_contents 属性就是实际值。
例如:
def outer(x):
def inner():
return x
return inner
<p>f = outer(42)
print(f.<strong>closure</strong>) # <tuple object at 0x...>
print(f.<strong>closure</strong>[0].cell_contents) # 42注意:__closure__ 为 None 表示不是闭包;空元组表示是闭包但没引用自由变量(比如只引用了全局变量)。
与 JavaScript 闭包的关键区别
JavaScript 中,每次循环迭代会创建新的词法环境,所以 let 声明的变量天然按次绑定;而 Python 没有块级作用域,for 循环不创建新作用域,变量 i 始终是同一个名字绑定到同一名字空间。因此 Python 的“变量绑定延迟求值”特性更明显,也更容易踩坑。










