1.闭包的介绍
我们前面已经学过了函数,我们知道当函数调用完,函数内定义的变量都销毁了,但是我们有时候需要保存函数内的这个变量,每次在这个变量的基础上完成一些列的操作,比如: 每次在这个变量的基础上和其它数字进行求和计算,那怎么办呢?这个时候就需要闭包来操作了。
闭包的定义:
在函数嵌套的前提下,内部函数使用了外部函数的变量,并且外部函数返回了内部函数,我们把这个使用外部函数变量的内部函数称为闭包。
2.闭包的构成条件
通过闭包的定义,我们可以得知闭包的形成条件:
1.在函数嵌套(函数里面在定义函数)的前提下
2.内部函数使用了外部函数的变量(还包括外部函数的参数)
3.外部函数返回内部函数
3.简单闭包的示例代码
# 定义一个外部函数
def func_out():
num1 = 2
# 定义一个内部函数
def func_inner(num2):
# 内部函数使用了外部函数的变量(num1)
result = num1 * num2
print("结果是:", result)
# 外部函数返回了内部函数,这里返回的内部函数就是闭包
return func_inner
# 创建闭包实例
f = func_out()
# 执行闭包
f(6)
f(8)
运行的结果是:
结果是:12
结果是:16
闭包执行结果的说明:
通过上面的输出结果可以看出闭包保存了外部函数内的变量num1,每次执行闭包都是在num1 = 2 基础上进行计算。
4.闭包的作用:
** 闭包可以保存外部函数内的变量,不会随着外部函数调用而销毁。**
需要注意的是:由于闭包引起了外部函数的变量,则外部函数的变量没有及时释放,消耗内存。
5.闭包的使用:
小面我们以一个小的案例来描述下闭包的使用,有这样的一个需求:
根据配置信息使用闭包实现不同人的对话信息。
例如:
老板:什么时候我们的产品能上线啊!!!
你:。。。。。。
分析:
首先,我们想到了python中有闭包这个功能可以是实现目前的这个需求,根据形成闭包的条件我们知道:第一步:我们需要定义外部函数接收不同配置信息参数,参数是人。第二步:需要定义内部函数接收对话信息参数。第三步:需要再定义函数里面把配置信息和对话信息进行拼接输出。
功能代码的实现:
def config_person(preson):
#内部函数
def say_info(info):
print(preson + ":" + infon)
return say_info
boss = config_preson("Boss")
boss("什么时候我们的产品能上线啊!!!")
you = config_preson(you)
you("。。。。。。")
上述案例表明:闭包还可以提高代码的可重复性,不在需要手动定义额外的功能函数
6.修改闭包内部使用的外部变量
在使用闭包的时候,我们如何能正确的修改外部变量呢?首先,展示一个错误实例:
# 定义一个外部函数
def func_out(num1):
# 定义一个内部函数
def func_inner(num2):
globals num1 # 这里本意想要修改外部num1的值,实际上是在内部函数定义了一个局部变量num1
num1 = 10
# 内部函数使用了外部函数的变量(num1)
result = num1 + num2
print("结果是:", result)
print(num1)
func_inner(1)
print(num1)
# 外部函数返回了内部函数,这里返回的内部函数就是闭包
return func_inner
# 创建闭包实例
f = func_out(1)
# 执行闭包
f(2)
所以,修改闭包使用的外部函数变量使用globals 是错误的,而真正修改使用的是nonlocal
正确实例:
# 定义一个外部函数
def func_out(num1):
# 定义一个内部函数
def func_inner(num2):
# 这里本意想要修改外部num1的值,实际上是在内部函数定义了一个局部变量num1
nonlocal num1 # 告诉解释器,此处使用的是 外部变量a
# 修改外部变量num1
num1 = 10
# 内部函数使用了外部函数的变量(num1)
result = num1 + num2
print("结果是:", result)
print(num1)
func_inner(1)
print(num1)
# 外部函数返回了内部函数,这里返回的内部函数就是闭包
return func_inner
# 创建闭包实例
f = func_out(1)
# 执行闭包
f(2)
总结:
1.当返回的内部函数使用了外部函数的变量就形成了闭包
2.闭包可以对外部函数的变量进行保存
3.实现闭包的标准格式:
# 外部函数
def test1(a):
b = 10
# 内部函数
def test2():
# 内部函数使用了外部函数的变量或者参数
print(a, b)
# 返回内部函数, 这里返回的内部函数就是闭包实例
return test2