Python闭包:函数记住状态的秘密

函数闭包 (Closure)

在 Python 中,闭包 指的是一个内部函数 (定义在另一个函数内部的函数)记住并访问其外部函数作用域中的变量 的能力,即使外部函数已经执行完毕

关键点:

  1. 嵌套函数: 闭包发生在嵌套函数的结构中。
  2. 访问外部变量: 内部函数可以访问(读取甚至修改)外部函数中定义的变量(包括参数)。
  3. 状态保持: 即使外部函数已经执行结束并返回了内部函数,内部函数仍然**"记住"** 了外部函数的局部作用域环境(包含那些外部变量)。

为什么需要闭包?

闭包提供了一种方式来封装数据(状态)和行为(函数) 在一起。它允许内部函数携带外部环境的信息,这在某些场景下非常有用:

  • 数据隐藏/封装: 外部函数的变量对于外部代码是"隐藏"的,只能通过返回的内部函数(闭包)来间接访问或修改。
  • 保持状态: 当需要函数记住某些状态信息时(比如计数器),闭包非常方便。
  • 实现装饰器: Python 装饰器的核心机制之一就是闭包。
  • 回调函数: 在事件处理或异步编程中,闭包可以将上下文信息传递给回调函数。

示例代码:

python 复制代码
def outer_function(msg):  # 外部函数
    message = msg  # 外部函数的局部变量

    def inner_function():  # 内部函数 (即将成为闭包)
        print(message)  # 内部函数访问外部函数的变量 `message`

    return inner_function  # 返回内部函数本身(注意:不是调用它)

# 创建闭包
my_closure = outer_function("你好,闭包!")

# 此时 outer_function 已经执行完毕,但...
my_closure()  # 输出:你好,闭包!

解释:

  1. outer_function 接受一个参数 msg,并将其赋值给局部变量 message
  2. outer_function 内部定义了 inner_function
  3. inner_function 访问了外部变量 message
  4. outer_function 返回inner_function 这个函数对象(没有执行它)。
  5. 调用 outer_function("你好,闭包!") 时,它执行完毕并返回了 inner_function。这个返回的 inner_function 函数对象被赋值给 my_closure
  6. 即使 outer_function 已经执行结束,其作用域本应消失,但当调用 my_closure() 时,它仍然能够打印出 "你好,闭包!"。这是因为 inner_function 作为一个闭包,记住了它被创建时所处的环境(包含 message = "你好,闭包!")。

另一个示例(计数器):

python 复制代码
def counter(start=0):
    count = start  # 外部函数变量,用于记录状态

    def increment():
        nonlocal count  # 声明 count 不是局部变量,而是来自外层作用域
        count += 1
        return count

    return increment

# 创建两个独立的计数器
counter1 = counter(10)
counter2 = counter()

print(counter1())  # 输出:11
print(counter1())  # 输出:12
print(counter2())  # 输出:1
print(counter2())  # 输出:2

解释:

  • counter 函数定义了变量 count 来保存计数值。
  • 内部函数 increment 可以访问并修改 count
  • counter 返回 increment 函数。
  • 每次调用 counter() 都会创建一个新的作用域 和一个新的 count 变量 ,因此 counter1counter2相互独立的计数器闭包,各自维护自己的状态。

总结:

闭包是 Python 中一个强大的特性,它允许内部函数"携带"其外部环境(变量),即使外部函数已经结束。这为数据封装、状态保持以及实现更高级的模式(如装饰器)提供了基础。理解闭包的关键在于认识到内部函数对其定义时所在作用域(词法作用域)的持久引用。

相关推荐
啊阿狸不会拉杆8 小时前
《机器学习导论》第 9 章-决策树
人工智能·python·算法·决策树·机器学习·数据挖掘·剪枝
Mr_Xuhhh8 小时前
C++11实现线程池
开发语言·c++·算法
喵手8 小时前
Python爬虫实战:城市停车收费标准自动化采集系统 - 让停车费透明化的技术实践(附CSV导出 + SQLite持久化存储)!
爬虫·python·爬虫实战·零基础python爬虫教学·城市停车收费标准·采集城市停车收费数据·采集停车数据csv文件导出
无水先生8 小时前
python函数的参数管理(01)*args和**kwargs
开发语言·python
py小王子8 小时前
dy评论数据爬取实战:基于DrissionPage的自动化采集方案
大数据·开发语言·python·毕业设计
Pyeako8 小时前
opencv计算机视觉--LBPH&EigenFace&FisherFace人脸识别
人工智能·python·opencv·计算机视觉·lbph·eigenface·fisherface
小陶的学习笔记8 小时前
python~基础
开发语言·python·学习
多恩Stone8 小时前
【3D AICG 系列-9】Trellis2 推理流程图超详细介绍
人工智能·python·算法·3d·aigc·流程图
ID_180079054738 小时前
Python结合淘宝关键词API进行商品价格监控与预警
服务器·数据库·python
lsx2024068 小时前
JavaScript 条件语句
开发语言