python闭包closure

在讨论闭包以前,先总结一下python的命名空间namespace,通常的语言都是经过namespace来识别名字标识,不管是变量,对象,函数等等。python划分3个名字空间层次,local:局部,标识为当前函数内,当前类内,好比局部变量。global:全局,标识当前模块,也就是当前文件,好比全局变量等。最后一类 built-in,内建,这个是做用域比较大,跨模块(文件)均可以标识,好比咱们自建的文件中,引用内建函数dir(), 这就是一个典型的例子,自建的文件中可以并无声明dir(),但因为dir()是built_in内建函数,故在全部文件或模块均可以引用。python

再来讨论闭包:百度上对闭包的定义是:闭包是指能够包含自由变量的代码块。定义简单明了,也有把自由变量成为环境变量的。紧接着定义什么是自由变量或环境变量:这些变量不是在这个代码块内或者任何全局上下文中定义的,而是在定义代码块的环境中定义(局部变量)。结合namespace的概念,自由变量的位置:既不在函数local范围内,也不在全局global范围内,而是在两个函数的做用域之间。举个例子:c++

 1 def numFunc(a, b):
 2     num = 100
 3     num2 = 200
 4     num3 = 300
 5     print('print s in numbunc', s)
 6 
 7     def addfunc(a, b):
 8         s = 'string in addfunc'
 9         x = num
10         y = num3
11         print('print s in addFunc', s)
12 
13     return addfunc

num, num2, num3, 这三个变量符合自由变量的描述,既不在局部local内,也再也不全局global内。而是在两个函数的做用域之间。闭包

因此我们说白了,闭包从形式上说,就是函数内嵌一个函数。可是还需知足两个条件,1.内层函数引用环境变量(自由变量)2外部函数返值为内部函数名(一个返回函数的高阶函数)函数

 

再来讲闭包有什么用? 闭包减小了参数的传递数量, 设计闭包是为了增长代码重复利用。还有的文章中说到,闭包是为了方便并行计算设计的,随着咱们深刻学习,会继续充实这篇文章,发觉闭包的更多做用。另外是否是跟装饰器有点像呢?其实本质上,decorator就是一个返回函数的高阶函数,闭包也是一个返回函数的高阶函数。从这个角度上说 装饰器也是闭包,闭包也是装饰器。区别在于装饰器但愿在代码运行期间动态增长功能,闭包但愿简化参数的调用提升代码利用率学习

 

闭包的特性 __closure__ui

__closure__是内部函数的一个属性,用来保存环境变量,用type()函数看一下,__closure__是一个tulple, 还以上边的代码为例,咱们看一下环境变量都包含什么,什么样的变量能够记录到环境变量中得以保存:

spa

 

从结果中咱们看到,num和num3被保存了下来,而num2没有被保存,缘由很简单,由于定义中,内部函数必须引用自由变量,num2没有被引用。设计

再来看一个例子,帮助咱们理解,解释器是如何保存环境变量到__closure__中来的,我把廖雪峰的例子作了修改:code

 1 def count():
 2     fs = []
 3     for i in range(1, 4):
 4         def f():
 5              return i*i 6  7     return f
 8 
 9 f = count()
10 x = None
11 x = f()12 pass

咱们单步跟进,发现,在每次循环中,f()只是声明,没有执行,因此每次循环,return i*i并无被执行。循环结束i = 3,在这时,count()函数返回,解释器将环境变量 i 保存到__closure__中去,i = 3,打扫好现场,count()的堆栈,上下文撤销对象

 

全部环境变量都是在函数闭包声明结束是完成初始化。下面咱们用一个例子来讲明:

第一步:   定义闭包以前,__closure__这个属性没有值

 

第二步:最关键的一步: 咱们发现,在闭包声明结束的时候,内部函数的还将变量就已经肯定下来了,不须要等到执行内部函数才肯定环境变量。

 

 

  

 

闭包的不习惯

在c/c++中好像没有闭包的概念,也没有环境变量(自由变量)的概念,因此函数包含函数,或者外部函数返回内部函数,会形成异常,由于内部函数返回,堆栈消失,全部内部变量都不存在了。而python的闭包设计是容许返回局部变量的,这给咱们这些从c/c++转过来的少年带来了很大的不适应,总感受要出大事,这时候想一想__closure__属性,python的闭包设计使推出内部函数后,没有回收内部函数的部分资源,而是做为环境变量保存下来了,慢慢习惯。

相关文章
相关标签/搜索