当前位置: 首页 > news >正文

Python 编程 - 闭包

在 Python 3 中,闭包(Closure) 是指在一个嵌套函数中,内部函数引用了外部函数的自由变量(即外部函数作用域中的变量),并且外部函数返回了内部函数本身。这样,即使外部函数执行完毕,内部函数依然可以记住并访问那些被引用的外部变量,形成了一种“闭包”环境。


闭包的三个必要条件

  1. 必须有一个嵌套函数(函数内部定义另一个函数)。
  2. 内部函数必须引用外部函数的变量(非全局变量)。
  3. 外部函数必须返回内部函数(或者把内部函数传给其他对象)。

最简单的闭包示例

def outer(x):def inner(y):return x + yreturn inneradd_5 = outer(5)       # 返回 inner 函数,并记住了 x=5
print(add_5(3))        # 输出 8
print(add_5(10))       # 输出 15

在这里,inner 引用了外部函数 outer 的局部变量 x。即使 outer 执行完毕,add_5 仍然持有 x=5 的值,这就是闭包。


如何查看闭包中捕获的变量

可以通过函数的 __closure__ 属性查看:

def outer(x):def inner(y):return x + yreturn innerf = outer(10)
print(f.__closure__)      # (<cell at 0x...: int object at 0x...>,)
print(f.__closure__[0].cell_contents)  # 10

闭包与 nonlocal 关键字

如果内部函数需要修改外部函数的变量(而不是仅仅读取),必须使用 nonlocal 声明,否则 Python 会认为你在内部函数中新建了一个局部变量。

def counter():count = 0def increment():nonlocal count   # 声明 count 不是局部变量count += 1return countreturn incrementc = counter()
print(c())  # 1
print(c())  # 2

如果没有 nonlocalcount += 1 会抛出 UnboundLocalError


闭包的常见应用场景

  • 装饰器(Decorator):闭包是实现装饰器的核心机制,装饰器接受函数,在内部包装并返回新函数。
  • 工厂函数:生成带有特定配置的函数(如上面的 outer)。
  • 回调函数:在事件驱动或异步编程中,携带额外状态。
  • 函数式编程:实现柯里化(Currying)、部分应用等。

注意事项

  1. 变量生命周期:闭包会延长外部变量的生命周期,它们不会在外部函数返回后立即释放,而是随闭包函数一起存在。如果大量使用闭包且不注意,可能导致内存泄漏。
  2. 循环中的闭包陷阱:在循环中创建闭包时,如果内部函数引用了循环变量,往往得到意外的结果(所有函数共享同一个变量)。解决方案是使用默认参数或 functools.partial
# 错误示例
funcs = []
for i in range(3):funcs.append(lambda: i)   # 所有 lambda 都引用同一个 i
print(funcs[0]())  # 2
print(funcs[1]())  # 2
print(funcs[2]())  # 2# 正确做法:用默认参数捕获当前值
funcs = []
for i in range(3):funcs.append(lambda i=i: i)  # 默认参数在定义时求值
print(funcs[0]())  # 0
print(funcs[1]())  # 1
print(funcs[2]())  # 2
  1. 性能:闭包比普通函数调用稍慢,因为涉及自由变量的访问(通过 LOAD_DEREF 指令),但在大多数场景下可忽略。

与 Python 2 的区别

  • Python 2 中,闭包只能读取外部变量,不能修改(没有 nonlocal)。Python 3 引入了 nonlocal,使得修改外层非全局变量成为可能。
  • Python 3 对闭包的实现更加清晰,__closure__ 属性统一。

总结

闭包是 Python 函数式编程的重要基石,理解它有助于掌握装饰器、高阶函数等高级特性。关键在于嵌套函数引用自由变量返回函数,并注意使用 nonlocal 来修改变量值。

http://www.jsqmd.com/news/1057178/

相关文章:

  • ZeroClaw:Windows本地AI指令网关实战指南
  • CentOS 8用户管理实战:systemd、SELinux与安全审计深度解析
  • Parsec VDD 0.45深度解析:虚拟显示驱动的技术内幕与实战指南
  • 亚洲EMBA排名前三客观测评与科学择校指南
  • 2026年浙江杭州行政诉讼律师推荐指南:从行政处罚到征收补偿全解析 - 本地品牌推荐
  • AI教材编写新利器!低查重AI写教材工具,高效打造专业教材!
  • 能量正则化神经MPC:提升全向飞行机器人控制精度的关键技术
  • 资本热捧具身大脑:融资提速、技术路线多元,泡沫与机遇并存?
  • 汽车MCU电源系统设计:MPC5775K多电压域与噪声管理实战
  • Qwen 3.6-27B:FP8量化+vLLM调度的边缘大模型工程实践
  • 医疗多模态大模型评估:MedImageEdu基准下的性能分析与挑战
  • Spring SpEL表达式注入漏洞:原理、审计与修复实战指南
  • 2026年浙江杭州合同纠纷律师推荐清单:5家专业团队实力解析 - 本地品牌推荐
  • BGU8053低噪声放大器设计:噪声系数与线性度平衡实战
  • LangGraph+Gradio实战:构建可调试可扩展的Agent系统
  • 生成式AI与酷儿艺术:数据伦理、算法偏见与社群抵抗的深层张力
  • 字符串连接的c++代码
  • 3个技巧让WE Learn网课学习效率提升300%:开源助手的智能解法
  • 本地Codex搭建实战:Ollama+Continue分层部署指南
  • 深圳搬家哪家强?2026年实测5家口碑公司,从起步价到附加费全拆解,拒绝坐地起价 - 从来都是英雄出少年
  • 003-费曼独立思考的底层哲学
  • 电动车托运哪个最安全?2026高保障平台实测推荐 - 快递物流资讯
  • 3分钟解决Windows 11界面不适:ExplorerPatcher完整指南
  • 构建Selenium持续测试流水线:从自动化脚本到工程化实践
  • 炉石传说脚本终极指南:5分钟快速上手的智能自动化对战工具
  • 两个小物件儿 ☜请点击这里可看全文
  • 2026无锡本土靠谱GEO SEO优化公司3家本土服务商实测:实测避坑,企业AI获客少走弯路 - wxxwlm
  • 武汉科谷技工学校2026年招生简章-城市轨道交通运输与管理专业怎么样?就业率/免学费/升学通道全解析 - 武汉中职最新信息发布
  • 钢结构稳定性分析的三种方法简介及区分
  • Jellyfin桌面客户端深度解析:专业级媒体播放器配置实战指南