关于闭包
简单来说,闭包就是函数内部定义的函数,被返回了出去并在外部调用。内部的函数存在外部作用域的引用就会导致闭包。
闭包会导致作用域链不被释放,作用域中的变量参数会被保存下来。最大的作用域全局作用域只有页面关闭才会释放。
变量查询是从当前函数的作用域链中从顶部依次向下查询。代码执行的环境和基本类型值存储在栈内存中,函数执行完,但是函数的私有作用域内有内容被栈外的变量还在使用的,栈内存就不能释放里面的基本值也就不会被释放。所以作用域链不被释放。
function a(){
var num = 100;
function b(){
num ++;
console.log(num);
}
return b;
}
var demo = a();
demo()
demo()
demo()
demo()
函数执行四次,结果分别为
可以看出每次执行改变的结果都被保存下来了。
作用域链不被释放,什么是作用域链?
作用域链:[[scopes]]中所存储的执行上下文的集合,以链接结构进行存储。
执行上下文:当函数运行时,会创建一个称为执行上下文的内部对象,一个执行上下文定义了函数运行时的环境,函数每次执行时都会创建一个独一无二的执行上下文,并且在函数执行完毕之后,会销毁掉函数所产生的执行上下文。
[[scopes]]中存储:[0:GO]或[0:AO,1:GO]或[0:AO,1:AO,2:GO]。第0位存放的永远是自己的上下文。
变量查询是从当前函数的作用域链中从顶部依次向下查询,就是从第0位向下查询。
预编译
- 全局预编译
a . 会生成一个(Gloabl Object) GO对象
b . 分析var变量声明,如果变量在GO对象中不存在,直接将变量名作为AO对象的属性名,值为undefined,如果已经存在,不做任何改变
c . 分析function函数,将函数名作为GO对象的属性名,值为函数体,如果已存在,直接覆盖 - 局部预编译
a . 函数在运行的一瞬间,会产生一个(Active Object) AO对象
b . 函数声明的参数(形参),成为AO对象的属性,属性值为实参的值,如果没有传递实参 属性值为undefined
c . 分析var变量声明,如果变量在AO对象中不存在,直接将变量名作为AO对象的属性名,值为undefined,如果已经存在,不做任何改变
d . 分析function函数,将函数名作为AO对象的属性名,值为函数体,如果已存在,直接覆盖。