别院牧志知识库 别院牧志知识库
首页
  • 基础

    • 全栈之路
    • 😎Awesome资源
  • 进阶

    • Python 工匠系列
    • 高阶知识点
  • 指南教程

    • Socket 编程
    • 异步编程
    • PEP 系列
  • 面试

    • Python 面试题
    • 2022 面试记录
    • 2021 面试记录
    • 2020 面试记录
    • 2019 面试记录
    • 数据库索引原理
  • 基金

    • 基金知识
    • 基金经理
  • 细读经典

    • 德隆-三个知道
    • 孔曼子-摊大饼理论
    • 配置者说-躺赢之路
    • 资水-建立自己的投资体系
    • 反脆弱
  • Git 参考手册
  • 提问的智慧
  • 分类
  • 标签
  • 归档
GitHub (opens new window)
首页
  • 基础

    • 全栈之路
    • 😎Awesome资源
  • 进阶

    • Python 工匠系列
    • 高阶知识点
  • 指南教程

    • Socket 编程
    • 异步编程
    • PEP 系列
  • 面试

    • Python 面试题
    • 2022 面试记录
    • 2021 面试记录
    • 2020 面试记录
    • 2019 面试记录
    • 数据库索引原理
  • 基金

    • 基金知识
    • 基金经理
  • 细读经典

    • 德隆-三个知道
    • 孔曼子-摊大饼理论
    • 配置者说-躺赢之路
    • 资水-建立自己的投资体系
    • 反脆弱
  • Git 参考手册
  • 提问的智慧
  • 分类
  • 标签
  • 归档
GitHub (opens new window)
  • 辨析

  • Sockets编程

  • Django

  • stackoverflow

  • Flask

  • 全栈之路

  • 面试

  • 代码片段

  • 异步编程

  • 😎Awesome资源

  • PEP

  • Python工匠系列

  • 高阶知识点

    • Python 中的 GIL
    • 关于 Python 闭包理解
      • 前言
      • 代码示例
      • 定义
      • __closure__ 属性和 cell 对象
      • 总结
      • 参考链接
      • 原文链接
    • Python 中的元类
    • Python 装饰器✨
    • Python 缓存机制与 functools.lru_cache
    • Python 受推崇的 super
    • Python 中的 MRO 与多继承
    • (译)Python 魔法方法指南
    • Python 的内存管理与垃圾回收机制
  • Python 学习资源待整理
  • 设计模式

  • 如何使用 Python 操作 RabbitMQ
  • 好“艹蛋”的 Python 呀!
  • FIFO | 待学清单📝
  • pip 安装及使用
  • 数据分析

  • 源码阅读计划

  • OOP

  • 关于 python 中的 setup.py
  • 并行分布式框架 Celery
  • 七种武器,让你的代码提高可维护性
  • 使用 pdb 调试 Python 代码
  • 每周一个 Python 标准库
  • 🐍Python
  • 高阶知识点
佚名
2020-08-06
目录

关于 Python 闭包理解

# 前言

闭包这个概念在 JavaScript 中讨论和使用得比较多,不过在 Python 中却不是那么显而易见,之所以说“不是那么”,是因为即使用到了,也没有注意到而已,比如定义一个 Decorator 时,就已经用到闭包了。网上对闭包的各种解释,感觉非常晦涩,在这里谈谈我的浅显认识:要形成闭包,首先得有一个嵌套的函数,即函数内部定义了另一个函数,闭包则是一个集合,它包括了外部函数的局部变量,这些局部变量在外部函数返回后也继续存在,并能被内部函数引用。

# 代码示例

这是个经常使用到的例子,定义一个函数 generate_power_func,它返回另一个函数,现在闭包形成的条件已经达到。

def generate_power_func(n):
    print("id(n): %X" % id(n))
    def nth_power(x):
        return x**n
    print("id(nth_power): %X" % id(nth_power))
    return nth_power
1
2
3
4
5
6

对于内部函数 nth_power,即使 generate_power_func 已经返回,它还是能引用到外部函数的局部变量 n,这种现象就称为闭包。具体运行一下:

>>> raised_to_4 = generate_power_func(4)
id(n): 246F770
id(nth_power): 2C090C8
>>> repr(raised_to_4)
'<function nth_power at 0x2c090c8>'
1
2
3
4
5

从结果可以看出,当 generate_power_func(4) 执行后, 创建并返回了 nth_power 这个函数对象,内存地址是 0x2C090C8,并且发现 raised_to_4 和它的内存地址相同,即 raised_to_4 只是这个函数对象的一个引用。先在全局命名空间中删除 generate_power_func,再试试会出现什么结果。

>>> del generate_power_func
>>> raised_to_4(2)
16
1
2
3

啊哈,居然没出现错误, nth_power 是怎么知道 n 的值是 4,而且现在 generate_power_func 甚至都不在这个命名空间了。对,这就是闭包的作用:外部函数的局部变量可以被内部函数引用,即使外部函数已经返回了。

# 定义

INFO

嵌套定义在非全局作用域中的函数,当它的外部函数被调用就会生成一个闭包。闭包中包含了这个子函数本身的代码与其依赖的外部变量的引用。

# __closure__ 属性和 cell 对象

现在知道闭包是怎么一回事了,那就到看看闭包到底是怎么回事的时候了。Python 中函数也是对象,所以函数也有很多属性,和闭包相关的就是 __closure__ 属性。__closure__ 属性定义的是一个包含 cell 对象的元组,其中元组中的每一个 cell 对象用来保存作用域中变量的值。

>>> raised_to_4.__closure__
(<cell at 0x2bf4ec0: int object at 0x246f770>,)
>>> type(raised_to_4.__closure__[0])
<type 'cell'>
>>> raised_to_4.__closure__[0].cell_contents
4
1
2
3
4
5
6

就如刚才所说,在 raised_to_4 的 __closure__ 属性中有外部函数变量 n 的引用,通过内存地址可以发现,引用的都是同一个 n。如果没有形成闭包,则 __closure__ 属性为 None。对于 Python 具体是如何实现闭包的,可以查看 Python 闭包详解 (opens new window),它通过分析 Python 字节码来讲述闭包的实现。

# 总结

面试题:为什么要用闭包? - SegmentFault 思否 (opens new window)

方便用户调用函数。不必为了维护繁杂的外部状态而烦恼。 例如 python,就把闭包玩出了很多花样:

  • 静态私有变量
  • 偏函数啦
  • 单参化
  • 装饰器

闭包特性有着非常多的作用,不过都是需要时才会不经意的用上,不要像使用设计模式一样去硬套这些法则。这篇文章按照自己的理解翻译自 Python Closures Explained (opens new window),可能和原文有些不同之处,如有疑惑,请查看原文。

附上一些参考资料。

# 参考链接

  • 浅显理解 Python 闭包 - I'm SErHo (opens new window)
  • 闭包的概念、形式与应用 (opens new window): 可以从其中了解闭包的应用
  • Python 闭包详解 (opens new window):从字节码出发了解 Python 闭包的实现机制
  • 理解 Python 闭包概念 - alpha_panda - 博客园 (opens new window)
  • 理解 Javascript 的闭包 (opens new window): 从 Javascript 的闭包中了解一些闭包特性,可以和 Python 作下对比
  • Why aren't python nested functions called closures? - StackOverflow (opens new window)

# 原文链接

浅显理解 Python 闭包 - I'm SErHo (opens new window)

编辑 (opens new window)
#Python#闭包
上次更新: 2023-03-17, 02:52:41
Python 中的 GIL
Python 中的元类

← Python 中的 GIL Python 中的元类→

最近更新
01
延米
03-23
02
装修名词
03-23
03
它山之石,可以攻玉,装修知识从哪学?
03-22
更多文章>
Theme by Vdoing | Copyright © 2019-2023 IMOYAO | 别院牧志
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式