对于已经对 闭包 或者 装饰器有必定概念的,能够直接经过右侧标题目录直接定位到相应段落查看所需的内容。html
装饰器(Decorator)相对简单,我们先介绍它:“装饰器的功能是将被装饰的函数看成参数传递给与装饰器对应的函数(名称相同的函数),并返回包装后的被装饰的函数”,听起来有点绕,不要紧,直接看示意图,其中 a 为与装饰器 @a 对应的函数, b 为装饰器修饰的函数,装饰器@a的做用是:python
简而言之:@a 就是将 b 传递给 a(),并返回新的 b = a(b)数组
栗子:闭包
上面使用@dobi来表示装饰器,其等同于:qinfeng = dobi(qinfeng)
所以装饰器本质上就是个语法糖,其做用为简化代码,以提升代码可读性,运行上段代码的结果为:函数
解析过程是这样子的:
1.python 解释器发现@dobi,就去调用与其对应的函数( dobi 函数)
2.dobi 函数调用前要指定一个参数,传入的就是@dobi下面修饰的函数,也就是 qinfeng()
3.dobi() 函数执行,调用 qinfeng(),qinfeng() 打印“dobi”spa
首先还得从基本概念提及,什么是闭包呢?来看下维基上的解释:
在计算机科学中,闭包(Closure)是词法闭包(Lexical Closure)的简称,是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即便已经离开了创造它的环境也不例外。因此,有另外一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。闭包在运行时能够有多个实例,不一样的引用环境和相同的函数组合能够产生不一样的实例。
....
上面提到了两个关键的地方: 自由变量 和 函数, 这两个关键稍后再说。仍是得在赘述下“闭包”的意思,望文知意,能够形象的把它理解为一个封闭的包裹,这个包裹就是一个函数,固然还有函数内部对应的逻辑,包裹里面的东西就是自由变量,自由变量能够在随着包裹处处游荡。固然还得有个前提,这个包裹是被建立出来的。
在经过Python的语言介绍一下,一个闭包就是你调用了一个函数A,这个函数A返回了一个函数B给你。这个返回的函数B就叫作闭包。你在调用函数A的时候传递的参数就是自由变量。
举个栗子:3d
def func(name): def inner_func(age): print 'name:', name, 'age:', age return inner_func bb = func('the5fire') bb(26) # >>> name: the5fire age: 26
这里面调用func的时候就产生了一个闭包——inner_func,而且该闭包持有自由变量——name,所以这也意味着,当函数func的生命周期结束以后,name这个变量依然存在,由于它被闭包引用了,因此不会被回收。code
另外再说一点,闭包并非Python中特有的概念,全部把函数作为一等公民的语言均有闭包的概念。不过像Java这样以class为一等公民的语言中也可使用闭包,只是它得用类或接口来实现。htm
在 python 的函数内,能够直接引用外部变量,但不能改写外部变量,所以若是在闭包中直接改写父函数的变量,就会发生错误:blog
在 python 2 中能够在函数内使用 global 语句,但全局变量在任何语言中都不被提倡,由于它很难控制,python 3 中引入了 nonlocal 语句解决了这个问题:
Nonlocal 与 global 的区别在于 nonlocal 语句会去搜寻本地变量与全局变量之间的变量,其会优先寻找层级关系与闭包做用域最近的外部变量。
上面已经简单演示了装饰器的功能,事实上,装饰器就是一种的闭包的应用,只不过其传递的是函数:
@makeitalic 装饰器将函数 hello 传递给函数 makeitalic,函数 makeitalic 执行完毕后返回被包装后的 hello 函数,而这个过程其实就是经过闭包实现的。@makebold 也是如此,只不过其传递的是 @makeitalic 装饰过的 hello 函数,所以最后的执行结果 <b>
在 <i>
外层,这个功能若是不用装饰器,其实就是显式的使用闭包:
闭包的最大特色是能够将父函数的变量与内部函数绑定,并返回绑定变量后的函数(也即闭包),此时即使生成闭包的环境(父函数)已经释放,闭包仍然存在,这个过程很像类(父函数)生成实例(闭包),不一样的是父函数只在调用时执行,执行完毕后其环境就会释放,而类则在文件执行时建立,通常程序执行完毕后做用域才释放,所以对一些须要重用的功能且不足以定义为类的行为,使用闭包会比使用类占用更少的资源,且更轻巧灵活,现举一例:假设咱们仅仅想打印出各种动物的叫声,分别以类和闭包来实现:
能够看到输出结果是彻底同样的,但显然类的实现相对繁琐,且这里只是想输出一下动物的叫声,定义一个 Animal 类未免小题大作,并且 voice 函数在执行完毕后,其做用域就已经释放,但 Animal 类及其实例 dog 的相应属性却一直贮存在内存中:
而这种占用对于实现该功能后,则是没有必要的。
除此以外,闭包还有不少其余功能,好比用于封装等,另外,闭包有效的减小了函数参数的数目,这对并行计算很是有价值,好比可让每台电脑负责一个函数,而后串起来,实现流水化的做业等。