包含python闭包函数优势的词条( 四 )


如果以内联方式使用块对象,则无需声明 。块对象声明语法与函数指针声明语法相似 , 但是块对象应使用脱字符(^)而非星号指针 (*) 。下面的代码声明一个aBlock变量,它标识一个需传入三个参数并具有float返回值的块 。
float (^aBlock)(const int*, int, float);
l 创建一个块
块使用脱字符(^)作为起始标志,使用分号作为结束标志 。下面的例子声明一个简单块,并且将其赋给之前声明的block变量(oneFrom) 。
int (^oneFrom)(int);
oneFrom = ^(int anInt) {
return anInt - 1;
};
微观世界
如 果要更加深入的了解闭包以及函数a和嵌套函数b的关系,我们需要引入另外几个概念:函数的执行环境(execution context)、活动对象(call object)、作用域(scope)、作用域链(scope chain) 。以函数a从定义到执行的过程为例阐述这几个概念 。
1、当定义函数a的时候,js解释器会将函数a的作用域链(scope chain)设置为定义a时a所在的“环境”,如果a是一个全局函数 , 则scope chain中只有window对象 。
2、当函数a执行的时候,a会进入相应的执行环境(execution context) 。
3、在创建执行环境的过程中,首先会为a添加一个scope属性,即a的作用域,其值就为第1步中的scope chain 。即a.scope=a的作用域链 。
4、然后执行环境会创建一个活动对象(call object) 。活动对象也是一个拥有属性的对象 , 但它不具有原型而且不能通过JavaScript代码直接访问 。创建完活动对象后,把活动对象添加到a的作用域链的最顶端 。此时a的作用域链包含了两个对象:a的活动对象和window对象 。
5、下一步是在活动对象上添加一个arguments属性,它保存着调用函数a时所传递的参数 。
6、最后把所有函数a的形参和内部的函数b的引用也添加到a的活动对象上 。在这一步中 , 完成了函数b的的定义,因此如同第3步,函数b的作用域链被设置为b所被定义的环境,即a的作用域 。
到此,整个函数a从定义到执行的步骤就完成了 。此时a返回函数b的引用给c,又函数b的作用域链包含了对函数a的活动对象的引用,也就是说b可以访问到a中定义的所有变量和函数 。函数b被c引用,函数b又依赖函数a,因此函数a在返回后不会被GC回收 。
当函数b执行的时候亦会像以上步骤一样 。因此 , 执行时b的作用域链包含了3个对象:b的活动对象、a的活动对象和window对象 , 如下图所示:
如图所示,当在函数b中访问一个变量的时候 , 搜索顺序是先搜索自身的活动对象,如果存在则返回,如果不存在将继续搜索函数a的活动对象,依 次查找,直到找到为止 。如果整个作用域链上都无法找到,则返回undefined 。如果函数b存在prototype原型对象,则在查找完自身的活动对象 后先查找自身的原型对象,再继续查找 。这就是Javascript中的变量查找机制 。
应用场景
1、保护函数内的变量安全 。以最开始的例子为例,函数a中i只有函数b才能访问 , 而无法通过其他途径访问到,因此保护了i的安全性 。
2、在内存中维持一个变量 。依然如前例 , 由于闭包,函数a中i的一直存在于内存中,因此每次执行c(),都会给i自加1 。
以上两点是闭包最基本的应用场景,很多经典案例都源于此 。
回收机制
在Javascript中,如果一个对象不再被引用,那么这个对象就会被GC回收 。如果两个对象互相引用,而不再被第3者所引用 , 那么这两个互相引用的对象也会被回收 。因为函数a被b引用 , b又被a外的c引用,这就是为什么函数a执行后不会被回收的原因 。

推荐阅读