JavaScript作用域

运行期上下文:当函数执行前一刻,会创建一个AO对象。一个AO定义了一个函数执行时的环境,函数每次执行时对应的AO对象都是独一无二的,所以多次调用一个函数会导致创建多个AO对象,当函数执行完毕,它产生的AO会被销毁。

function test(){}
test();
test();//虽然两次调用产生的AO内容一模一样,但是还是属于两个独一无二的AO

[[scope]]:每个JavaScript函数都是一个对象,对象中有些属性我们可以访问,有些不可以,它只供JavaScript引擎存取,[[scope]]就是其中一个,它存储了运行期上下文集合,这个集合呈链式连接,我们叫它作用域链。

查找变量:从作用域链顶端依次向下查找。

例:

​
function a(){

}
var glob = 100;
a();

​

首先a函数在全局定义,生成一个GO,它存在作用域链的顶端 [[scope]][0](这里可以把作用域想象成一个类似栈的结构)。

然后a();执行前一刻,对函数进行预编译生成一个AO,这个AO会被压入作用域链的顶端,而GO则被AO压下去。

至此,[[scope]][0] -->AO        [[scope]][1]-->GO

所以,当函数内部执行时,它总是先去栈顶找,即自己的AO里寻找变量,AO中没有,才会去GO中寻找。

这就是所说的查找变量是从作用域链顶端依次向下查找。

别急,这只是一个开始。

 

让我们来看一个更复杂的例子:

​
function a () {
      function b () {

                var b =234;

                  }
                var a = 123;
                b();
}
                var glob =100;
                a();

​

1.首先,a函数在全局定义,a会生成一个GO存在作用域链顶端,即[[scope]][0]==aGO,

2.然后,由于a(); a在执行前一刻,a函数进行预编译,产生了一个AO,把它压入作用域链顶端,即[[scope]][0]==aAO,[[scope]][1]==aGO,

3.a函数开始执行,是不是产生了一个b函数的定义呢?b在a里面,a执行才会产生b定义哦。

4.b既然是一个函数,那么它就得有自己的作用域,b的环境是a给的,b是站在a的肩膀上看世界的,那么b便直接引用a的AO,GO(特别注意:此处b引用过来的AO,GO和a作用域链里的AO,GO是完全相同的,就是一个东西,是引用,不是拷贝,拷贝就是两个独一无二的东西长得一模一样)

5.然后紧接着b执行,b执行前,产生一个自己的独一无二的AO,压入栈顶,至此,

b的作用域链中:[[scope]][0]==bAO,

                          [[scope]][1]==aAO,

                          [[scope]][2]==aGO,

6.最后b执行完毕,同时a也执行完毕(因为b();是a函数的最后一句),a,b的AO被销毁,等待下一次调用时,重新生成另一个独一无二的AO。

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值