Python基础 3.9 闭包与装饰器

3.9 闭包 与 装饰器3.9.1 闭包1. 简单闭包1.闭包的介绍我们前面已经学过了函数,我们知道当函数调用完,函数内定义的变量都销毁了,但是我们有时候需要保存函数内的这个变量,每次在这个变量的基础上完成一些列的操作,比如: 每次在这个变量的基础上和其它数字进行求和计算,那怎么办呢?我们就可以通过咱们今天学习的闭包来解决这个需求。闭包的定义:在函数嵌套的前提下,内部函数使用了外部函数的变量,并且外部函数返回了内部函数,我们把这个使用外部函数变量的内部函数称为闭包。2.闭包的构成条件通过闭
摘要由CSDN通过智能技术生成

3.9 闭包 与 装饰器

3.9.1 闭包

1. 简单闭包

1.闭包的介绍

我们前面已经学过了函数,我们知道当函数调用完,函数内定义的变量都销毁了,但是我们有时候需要保存函数内的这个变量,每次在这个变量的基础上完成一些列的操作,比如: 每次在这个变量的基础上和其它数字进行求和计算,那怎么办呢?

我们就可以通过咱们今天学习的闭包来解决这个需求。

闭包的定义:

在函数嵌套的前提下,内部函数使用了外部函数的变量,并且外部函数返回了内部函数,我们把这个使用外部函数变量的内部函数称为闭包

2.闭包的构成条件

通过闭包的定义,我们可以得知闭包的形成条件:

  1. 在函数嵌套(函数里面再定义函数)的前提下
  2. 内部函数使用了外部函数的变量(还包括外部函数的参数)
  3. 外部函数返回了内部函数
3.简单闭包的示例代码
# 定义一个外部函数
def func_out(num1):
  	# num1 的引用计数+1
    # 定义一个内部函数
    def func_inner(num2):
        # 内部函数使用了外部函数的变量(num1)
        result = num1 + num2
        # num1 的引用计数+1
        print("结果是:", result)
    # 外部函数返回了内部函数,这里返回的内部函数就是闭包
    return func_inner
  	# num1 的引用计数-1
    # num1 的引用计数 +1+1-1 = 1

# 创建闭包实例 函数本身是一个类 我们创造了一个函数对象 闭包对象
# 函数是对象 也是变量,所以我们能引用给其他变量
f = func_out(1)
# f = func_inner() 其中num1 = 1
# 执行闭包 调用函数
f(2)
# f(2) = func_inner(2)
f(3)
# f(3) = func_inner(3)


# 那如何销毁num1和f
# 第一个方法 删除引用 使得其引用计数-1=0 被解释器自动销毁
del f
# 第二个方法 程序结束,所有变量全部销毁

运行结果:

结果是: 3
结果是: 4

闭包执行结果的说明:

  • 这段代码中函数func_inner就是闭包。
  • 通过上面的输出结果可以看出闭包保存了外部函数内的变量num1,每次执行闭包都是在num1 = 1 基础上进行计算。
  • Python中变量销毁,由解释器自动完成,解释器里有个自动垃圾回收机制,只要变量的引用计数不为0,就不会被销毁。
  • 一个函数完成会使引用计数为-1,而在闭包里,因为嵌套函数引用了外部函数的变量,导致其引用计数+1,加上函数调用的时候引用计数+1,使得变量引用计数即使函数结束后依旧不为0,不会被销毁。
  • 如果我们要销毁闭包和闭包引用的变量,可以使用del销毁闭包对象和变量,或者可以等程序结束后会自动销毁所有变量。

拓展 引用计数

import sys
# 定义一个变量a
a = [1, 2, 3, 4, 5]
print(sys.getrefcount(a))
# 因为传参引入了a 所以返回2
b = a
print(sys.getrefcount(a))
# 多了一个b引向a 计数+1 =3
c = b
print(sys.getrefcount(a))
print(sys.getrefcount(b))
# a多了一个引入c b也多了一个引入c 结果均为+1 =4
d = 1
print(sys.getrefcount(d))
# 常用变量在Python里是经常引用的 我们之前说过Python里会有变量共享,这些常用的变量引用的都是同一个地址里的内容

2. 闭包使用案例

1. 实现步骤说明
  1. 定义外部函数接收不同的配置信息参数,参数是人名
  2. 定义内部函数接收对话信息参数
  3. 在内部函数里面把配置信息和对话信息进行拼接输出
2. 功能代码的实现
def config_name(name):
  	# 定义了闭包say_info
    def say_info(msg):
      	# 使用了外部函数变量name
        print(name, ': ', msg)
    return say_info

# 创造了闭包对象tom
tom = config_name('Tom')
# 打印出tom的地址
print("tom的地址:", hex(id(tom)))
# 创造了新的闭包对象jerry 
jerry = config_name("Jerry")
# 打印出jerry的地址 是新的 不同于tom
print("jerry的地址:", hex(id(jerry)))
# 可以分别执行 毫不冲突
tom("hello")
jerry("hi")

运行结果:

tom的地址: 0x7fdfa56d3430
jerry的地址: 0x7fdfa56f21f0
Tom :  hello
Jerry :  hi

闭包案例说明:

  • 闭包还可以提高代码的可重用性,不需要再手动定义额外的功能函数。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值