闭包

一、什么是闭包

1、如果在一个内部函数中对外部函数作用域(非全局作用域)的变量进行引用,那么内部函数就会被称为闭包。也就是, 在一个外函数中定义了一个内函数,内函数里运用了外函数的临时变量,并且外函数的返回值是内函数的引用。这样就构成了一个闭包。

2、闭包需要满足以下3个条件:

1)存在两个嵌套关系的函数中,并且闭包是内部函数;

2)内部函数引用了外部函数的变量(自由变量);

3)外部函数会把内部函数的函数名称返回。

3、示例
在这里插入图片描述
解释:程序从上而下开始执行,先执行(1),紧接着执行(2),def inner() 到 return inner()都属于内部函数,相当于闭包,因此,暂时先不执行,直接跳到(3),开始执行(4),到(5)时才开始执行闭包函数的内容。

4、闭包的实例

#outer是外部函数 a和b都是外函数的临时变量
  def outer( a ):
      b = 10
      # inner是内函数
      def inner():
          #在内函数中 用到了外函数的临时变量
          print(a+b)
      # 外函数的返回值是内函数的引用
     return inner
 
 if __name__ == '__main__':
     # 在这里我们调用外函数传入参数5
     #此时外函数两个临时变量 a是5 b是10 ,并创建了内函数,然后把内函数的引用返回存给了demo
     # 外函数结束的时候发现内部函数将会用到自己的临时变量,这两个临时变量就不会释放,会绑定给这个内部函数
     demo = outer(5)
     # 我们调用内部函数,看一看内部函数是不是能使用外部函数的临时变量
     # demo存了外函数的返回值,也就是inner函数的引用,这里相当于执行inner函数
     demo() # 15
     demo2 = outer(7)
     demo2()#17

可以如此理解:

1)外函数返回了内函数的引用

当我们进行a=1的时候,实际上在内存当中有一个地方存了值1,然后用a这个变量名存了1所在内存位置的引用。引用就好像c语言里的指针,大家可以把引用理解成地址。a只不过是一个变量名字,a里面存的是1这个数值所在的地址,就是a里面存了数值1的引用。

对于闭包,在外函数outer中 最后return inner,我们在调用外函数 demo = outer() 的时候,outer返回了inner,inner是一个函数的引用,这个引用被存入了demo中。所以接下来我们再进行demo() 的时候,相当于运行了inner函数。

2)外函数把临时变量绑定给内函数

内部函数只定义了一次,我们发现调用的时候,内部函数是能识别外函数的临时变量是不一样的。python中一切都是对象,虽然函数我们只定义了一次,但是外函数在运行的时候,实际上是按照里面代码执行的,外函数里创建了一个函数,我们每次调用外函数,它都创建一个内函数,虽然代码一样,但是却创建了不同的对象,并且把每次传入的临时变量数值绑定给内函数,再把内函数引用返回。虽然内函数代码是一样的,但其实,我们每次调用外函数,都返回不同的实例对象的引用,他们的功能是一样的,但是它们实际上不是同一个函数对象。

二、如何在闭包中内函数修改外函数局部变量

1、在基本的python语法当中,一个函数可以随意读取全局数据,但是要修改全局数据的时候有两种方法:
1 global 声明全局变量
2 全局变量是可变类型数据的时候可以修改

在闭包内函数也是类似的情况。在内函数中想修改闭包变量(外函数绑定给内函数的局部变量)的时候:

1 在python3中,可以用nonlocal 关键字声明 一个变量, 表示这个变量不是局部变量空间的变量,需要向上一层变量空间找这个变量。

2 在python2中,没有nonlocal这个关键字,我们可以把闭包变量改成可变类型数据进行修改,比如列表。

2、修改闭包变量的实例

#outer是外部函数 a和b都是外函数的临时变量
 def outer( a ):
      b = 10  # a和b都是闭包变量
      c = [a] #这里对应修改闭包变量的方法2
      # inner是内函数
      def inner():
          #内函数中想修改闭包变量
          # 方法1 nonlocal关键字声明
         nonlocal  b
         b+=1
         # 方法二,把闭包变量修改成可变数据类型 比如列表
         c[0] += 1
         print(c[0])
         print(b)
     # 外函数的返回值是内函数的引用
     return inner
 
 if __name__ == '__main__': 
     demo = outer(5)
     demo() # 6  11

从上面代码中我们能看出来,在内函数中,分别对闭包变量进行了修改,打印出来的结果也确实是修改之后的结果。以上两种方法就是内函数修改闭包变量的方法。

注意:使用闭包的过程中,一旦外函数被调用一次返回了内函数的引用,虽然每次调用内函数,是开启一个函数执行过后消亡,但是闭包变量实际上只有一份,每次开启内函数都在使用同一份闭包变量

三、闭包的作用

1、装饰器!!!装饰器是做什么的??其中一个应用就是,我们工作中写了一个登录功能,我们想统计这个功能执行花了多长时间,我们可以用装饰器装饰这个登录模块,装饰器帮我们完成登录函数执行之前和之后取时间。

2、面向对象!!!经历了上面的分析,我们发现外函数的临时变量送给了内函数。大家回想一下类对象的情况,对象有好多类似的属性和方法,所以我们创建类,用类创建出来的对象都具有相同的属性方法。闭包也是实现面向对象的方法之一。在python当中虽然我们不这样用,在其他编程语言入比如avaScript中,经常用闭包来实现面向对象编程

3、实现单利模式!! 其实这也是装饰器的应用。单利模式毕竟比较高大。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值