闭包是什么?
function createComp(){
let str = " 我是一个值"
return function(value){
let another = str + value
return another
}
}
let res = createComp()
let aa = res("133")
当createComp被调用时,会创建一个执行环境
,以及相应的作用域链
。并把作用域链赋值给一个内部属性[[scope]]。
然后使用this, arguments 和其他命名参数来初始化函数的活动对象(activation object)
在作用域链中,外部函数的活动对象始终处于第二位。外部函数的外部函数活动对象处于第三位…直至作用域链终点的全局执行环境
。
全局环境的变量对象始终存在。而局部活动对象,则只有在函数执行的过程中存在。
在创建createComp()函数时,会创建一个预先包含全局变量对象的作用域链,这个作用域链被保存在内部的[[scope]]属性中。当调用createComp()时,会为函数创建一个执行环境,然后通过复制[[scope]]的作用域链,又构建起了执行环境的作用于链。当执行createComp()时,会把createComp的活动对象推入这个执行环境作用域链的前端。
在函数中访问一个变量时,它就会沿着作用域链中搜索具有相同名字的变量。当函数执行完毕后,该函数的活动对象就会被销毁,内存中仅保存全局作用域。但是,闭包就特殊了~
在createComp 这个函数中又定义了一个匿名函数,function(){…}。这个匿名函数从createComp中被返回后,它的作用域链被初始化为包含createComp函数的活动对象和全局变量对象。这样匿名函数就可以访问在createComp中的的所有变量。按照原先的规则,createComp的活动对象在这个函数执行完毕后应该被销毁,但是由于被返回到外部的匿名函数引用着,所以createComp的活动对象就不会被销毁。(虽然,createComp的执行环境被销毁了,但是它的活动对象被保留下来了!)。直到匿名函数被销毁后,createComp的活动对象才会被销毁。