-
Notifications
You must be signed in to change notification settings - Fork 4.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
JavaScript深入之变量对象 #5
Comments
Arguments对象是什么 - -。 |
引用《JavaScript权威指南》回答你哈:调用函数时,会为其创建一个Arguments对象,并自动初始化局部变量arguments,指代该Arguments对象。所有作为参数传入的值都会成为Arguments对象的数组元素。 |
VO 和 AO 到底是什么关系。 |
未进入执行阶段之前,变量对象(VO)中的属性都不能访问!但是进入执行阶段之后,变量对象(VO)转变为了活动对象(AO),里面的属性都能被访问了,然后开始进行执行阶段的操作。 它们其实都是同一个对象,只是处于执行上下文的不同生命周期。@jDragonV |
@jawil 非常感谢回答,一语中的。 |
@mqyqingfeng 楼主,有幸拜读你的深入系列,收获颇多,但也存在一些疑问。比如变量对象留给我们的思考题的第二题,按照你的写法: console.log(foo);
function foo(){
console.log("foo");
}
var foo = 1; // 打印函数 但个人觉得这句“这是因为在进入执行上下文时,首先会处理函数声明,其次会处理变量声明,如果如果变量名称跟已经声明的形式参数或函数相同,则变量声明不会干扰已经存在的这类属性。”解释得有点欠完整,如果我把代码改写成下面这样: var foo = 1;
console.log(foo);
function foo(){
console.log("foo");
}; 这次打印结果就是“1”; 所以我觉得这么解释比较好: 进入执行上下文时,首先会处理函数声明,首先会处理函数声明,其次会处理变量声明,如果如果变量名称跟已经声明的形式参数或函数相同,则变量声明不会干扰已经存在的这类属性。 进入代码执行阶段,先执行console.log(foo),此时foo是函数的应用,再执行var foo = 1;将foo赋值为1,而在我改写的例子里中,先执行var foo = 1;再执行console.log(foo),所以打印1。我觉得加上代码执行阶段会更清晰,哈哈哈 |
一个执行上下文的生命周期可以分为两个阶段。
都没有错,博主讲的主要是针对变量对象,而变量对象的创建是在EC(执行上下文)的创建阶段,所以侧重点主要是EC的生命周期的第一个阶段,我觉得再执行var foo = 1这句话有点不妥,应该是给foo赋值,应该是执行foo=1这个操作,因为在EC创建阶段var已经被扫描了一遍。 |
是的,显然你的说法更严谨,也符合分析的过程! 学习了@jawil |
@jawil 哈哈,十分感谢回答~~~ @alexzhao8326 这道题应该是因为没有分成两个阶段来讲,所以让你觉得分析得不是很完整吧。我在写的时候,觉得毕竟是思考题,讲清楚问题的关键点即可,所以也没有给出完整的分析。如果你看完前面的内容,相信你一定能明白结果为什么会是这样,对于你修改后的例子,相信你也能解释的了。当然了,学习时严谨的态度还是要有的,感谢指出,o( ̄▽ ̄)d |
@wedaren 进入执行上下文时,初始化的规则如下,从上到下就是一种顺序: |
执行上下文的时候:
然后再执行了 foo = 1 的操作,修改变量对象的 foo 属性值
执行代码 console.log(foo) 的结果: 1 |
@zuoyi615 感谢写下自己的分析过程,如果这段代码是在全局环境下执行的,变量对象应该用 VO 表示,此时也没有 arguments 属性 |
@zuoyi615 o( ̄▽ ̄)d |
@jawil ,你说的有一点误差,AO 实际上是包含了 VO 的。因为除了 VO 之外,AO 还包含函数的 parameters,以及 arguments 这个特殊对象。也就是说 AO 的确是在进入到执行阶段的时候被激活,但是激活的除了 VO 之外,还包括函数执行时传入的参数和 arguments 这个特殊对象。 |
@oakland 非常感谢补充~~~ 这一点我也没有注意到~ o( ̄▽ ̄)d |
是w3school 不是W3C school |
@ckclark 哎呀呀,我犯了一个严重的错误,非常感谢指出~o( ̄▽ ̄)d |
思考题第二题:
解:
因为JavaScript的函数提升特性,将代码等量变换为:(1)
又因为JavaScript的变量提升特性,将代码等量变换为:(2)
开始创建对应的
当javaScript扫描到
执行代码
解答完毕.
|
另外,以上两处代码得出的结论一样,说明:
大家知道的微微一笑就好了:) |
根据你们的讨论,关于这一段代码的实现,
执行结果是函数和1,我可以这样理解么?
望不吝赐教! |
变量对象跟词法环境可不可以理解为是同一种东西? |
1 similar comment
变量对象跟词法环境可不可以理解为是同一种东西? |
这篇文章是es3标准的解释,现在es6标准来说,执行上下文包括:词法环境、变量环境、this。 |
非常感谢 |
您好,来信已收到,我会尽快给你回复。
|
中文译文可看: 【译】理解 Javascript 执行上下文和执行栈 |
引用权威指南里的:如果没有使用严格模式并给一个未声明的变量赋值的话,JS会自动创建一个全局变量。 |
内容很好, 评价里的内容也很好! 受益匪浅! |
您好,来信已收到,我会尽快给你回复。
|
感谢博主,我们是否可以理解AO是先进行变量以及函数的提升,然后再执行代码。 |
是个好问题,我也觉得这里有点不妥,目前也没找到合理的解释。 |
@Vsnoy let不挂载到global,但是会挂载跟global同样层级的Script。仍然可以通过AO找到。
|
思考题2: |
AO = { |
这里补充《你不知道的JS上卷》中的一段话,感觉可以作为更好地理解执行上下文的补充。 关于JS引擎和编译器如何理解下面这行变量赋值。
下面我们将var a = 2;分解 编译器首先会将这段程序分解成词法单元,然后将词法单元解析成一个树结构。但是当编译器开始进行代码生成时,它对这段程序的处理方式会和预期的有所不同。可以合理地假设编译器所产生的代码能够用下面的伪代码进行概括:“为一个变量分配内存,将其命名为a,然后将值2保存进这个变量。”然而,这并不完全正确。事实上编译器会进行如下处理。 1.遇到var a,编译器会询问作用域是否已经有一个该名称的变量存在于同一个作用域的集合中。如果是,编译器会忽略该声明,继续进行编译;否则它会要求作用域在当前作用域的集合中声明一个新的变量,并命名为a。 2.接下来编译器会为引擎生成运行时所需的代码,这些代码被用来处理a = 2这个赋值操作。引擎运行时会首先询问作用域,在当前的作用域集合中是否存在一个叫作a的变量。如果是,引擎就会使用这个变量;如果否,引擎会继续查找该变量(查看1.3节)。如果引擎最终找到了a变量,就会将2赋值给它。否则引擎就会举手示意并抛出一个异常! 总结:变量的赋值操作会执行两个动作,首先编译器会在当前作用域中声明一个变量(如果之前没有声明过),然后在运行时引擎会在作用域中查找该变量,如果能够找到就会对它赋值。 从上面这段话来看,个人认为可以把执行过程这样解释:
|
您好,来信已收到,我会尽快给你回复。
|
在进入执行上下文时会给变量对象添加形参、函数声明、变量声明等初始的属性值 关于这段说法和前文的代码有个疑问请教 文中片段「在进入执行上下文的时候」 |
参照
函数传入实参时,变量对象的a就已经被赋予实际值了 |
您好,来信已收到,我会尽快给你回复。
|
简而言之 |
您好,来信已收到,我会尽快给你回复。
|
“当第二段执行 console 的时候,全局对象已经被赋予了 a 属性,这时候就可以从全局找到 a 的值,所以会打印 1。” |
您好,来信已收到,我会尽快给你回复。
|
function foo(a) { } foo("name"); 有个疑问,在执行阶段,形参被赋值时应该会覆盖解析阶段的值啊,此时打印出来的应该是name,而不应该是Function a啊 |
前言
在上篇《JavaScript深入之执行上下文栈》中讲到,当 JavaScript 代码执行一段可执行代码(executable code)时,会创建对应的执行上下文(execution context)。
对于每个执行上下文,都有三个重要属性:
今天重点讲讲创建变量对象的过程。
变量对象
变量对象是与执行上下文相关的数据作用域,存储了在上下文中定义的变量和函数声明。
因为不同执行上下文下的变量对象稍有不同,所以我们来聊聊全局上下文下的变量对象和函数上下文下的变量对象。
全局上下文
我们先了解一个概念,叫全局对象。在 W3School 中也有介绍:
如果看的不是很懂的话,容我再来介绍下全局对象:
1.可以通过 this 引用,在客户端 JavaScript 中,全局对象就是 Window 对象。
2.全局对象是由 Object 构造函数实例化的一个对象。
3.预定义了一堆,嗯,一大堆函数和属性。
4.作为全局变量的宿主。
5.客户端 JavaScript 中,全局对象有 window 属性指向自身。
花了一个大篇幅介绍全局对象,其实就想说:
全局上下文中的变量对象就是全局对象呐!
函数上下文
在函数上下文中,我们用活动对象(activation object, AO)来表示变量对象。
活动对象和变量对象其实是一个东西,只是变量对象是规范上的或者说是引擎实现上的,不可在 JavaScript 环境中访问,只有到当进入一个执行上下文中,这个执行上下文的变量对象才会被激活,所以才叫 activation object 呐,而只有被激活的变量对象,也就是活动对象上的各种属性才能被访问。
活动对象是在进入函数上下文时刻被创建的,它通过函数的 arguments 属性初始化。arguments 属性值是 Arguments 对象。
执行过程
执行上下文的代码会分成两个阶段进行处理:分析和执行,我们也可以叫做:
进入执行上下文
当进入执行上下文时,这时候还没有执行代码,
变量对象会包括:
函数的所有形参 (如果是函数上下文)
函数声明
变量声明
举个例子:
在进入执行上下文后,这时候的 AO 是:
代码执行
在代码执行阶段,会顺序执行代码,根据代码,修改变量对象的值
还是上面的例子,当代码执行完后,这时候的 AO 是:
到这里变量对象的创建过程就介绍完了,让我们简洁的总结我们上述所说:
全局上下文的变量对象初始化是全局对象
函数上下文的变量对象初始化只包括 Arguments 对象
在进入执行上下文时会给变量对象添加形参、函数声明、变量声明等初始的属性值
在代码执行阶段,会再次修改变量对象的属性值
思考题
最后让我们看几个例子:
1.第一题
第一段会报错:
Uncaught ReferenceError: a is not defined
。第二段会打印:
1
。这是因为函数中的 "a" 并没有通过 var 关键字声明,所有不会被存放在 AO 中。
第一段执行 console 的时候, AO 的值是:
没有 a 的值,然后就会到全局去找,全局也没有,所以会报错。
当第二段执行 console 的时候,全局对象已经被赋予了 a 属性,这时候就可以从全局找到 a 的值,所以会打印 1。
2.第二题
会打印函数,而不是 undefined 。
这是因为在进入执行上下文时,首先会处理函数声明,其次会处理变量声明,如果变量名称跟已经声明的形式参数或函数相同,则变量声明不会干扰已经存在的这类属性。
下一篇文章
《JavaScript深入之作用域链》
本文相关链接
《JavaScript深入之执行上下文栈》
深入系列
JavaScript深入系列目录地址:https://github.com/mqyqingfeng/Blog。
JavaScript深入系列预计写十五篇左右,旨在帮大家捋顺JavaScript底层知识,重点讲解如原型、作用域、执行上下文、变量对象、this、闭包、按值传递、call、apply、bind、new、继承等难点概念。
如果有错误或者不严谨的地方,请务必给予指正,十分感谢。如果喜欢或者有所启发,欢迎star,对作者也是一种鼓励。
The text was updated successfully, but these errors were encountered: