JavaScript深切之变量对象

JavaScript深切系列第四篇,详细解说实行高低文中的变量对象与运动对象。全局高低文下的变量对象是什么?函数高低文下的运动对象是怎样剖析和实行的?另有两个思考题帮你加深印象,快来看看吧!

媒介

在上篇《JavaScript深切之实行高低文栈》中讲到,当 JavaScript 代码实行一段可实行代码(executable code)时,会建立对应的实行高低文(execution context)。

关于每一个实行高低文,都有三个主要属性:

  • 变量对象(Variable object,VO)

  • 作用域链(Scope chain)

  • this

本日重点讲讲建立变量对象的历程。

变量对象

变量对象是与实行高低文相干的数据作用域,存储了在高低文中定义的变量和函数声明。

由于差别实行高低文下的变量对象稍有差别,所以我们来聊聊全局高低文下的变量对象和函数高低文下的变量对象。

全局高低文

我们先相识一个观点,叫全局对象。在 W3C school 中也有引见:

全局对象是预定义的对象,作为 JavaScript 的全局函数和全局属性的占位符。经由过程运用全局对象,能够接见一切其他一切预定义的对象、函数和属性。

在顶层 JavaScript 代码中,能够用关键字 this 援用全局对象。由于全局对象是作用域链的头,这意味着一切非限定性的变量和函数名都邑作为该对象的属性来查询。

比方,当JavaScript 代码援用 parseInt() 函数时,它援用的是全局对象的 parseInt 属性。全局对象是作用域链的头,还意味着在顶层 JavaScript 代码中声明的一切变量都将成为全局对象的属性。

假如看的不是很懂的话,容我再来引见下全局对象:

1.能够经由过程 this 援用,在客户端 JavaScript 中,全局对象就是 Window 对象。

console.log(this);

2.全局对象是由 Object 组织函数实例化的一个对象。

console.log(this instanceof Object);

3.预定义了一堆,嗯,一大堆函数和属性。

// 都能见效
console.log(Math.random());
console.log(this.Math.random());

4.作为全局变量的宿主。

var a = 1;
console.log(this.a);

5.客户端 JavaScript 中,全局对象有 window 属性指向本身。

var a = 1;
console.log(window.a);

this.window.b = 2;
console.log(this.b);

花了一个大篇幅引见全局对象,实在就想说:

全局高低文中的变量对象就是全局对象呐!

函数高低文

在函数高低文中,我们用运动对象(activation object, AO)来示意变量对象。

运动对象和变量对象实际上是一个东西,只是变量对象是范例上的或许说是引擎完成上的,不可在 JavaScript 环境中接见,只要到当进入一个实行高低文中,这个实行高低文的变量对象才会被激活,所以才叫 activation object 呐,而只要被激活的变量对象,也就是运动对象上的种种属性才被接见。

运动对象是在进入函数高低文时刻被建立的,它经由过程函数的 arguments 属性初始化。arguments 属性值是 Arguments 对象。

实行历程

实行高低文的代码会分红两个阶段举行处置惩罚:剖析和实行,我们也能够叫做:

  1. 进入实行高低文

  2. 代码实行

进入实行高低文

当进入实行高低文时,这时刻还没有实行代码,

变量对象会包含:

  1. 函数的一切形参 (假如是函数高低文)

    • 由称号和对应值构成的一个变量对象的属性被建立

    • 没有实参,属性值设为 undefined

  2. 函数声明

    • 由称号和对应值(函数对象(function-object))构成一个变量对象的属性被建立

    • 假如变量对象已存在雷同称号的属性,则完整替代这个属性

  3. 变量声明

    • 由称号和对应值(undefined)构成一个变量对象的属性被建立;

    • 假如变量称号跟已声明的形式参数或函数雷同,则变量声明不会滋扰已存在的这类属性

举个例子:

function foo(a) {
  var b = 2;
  function c() {}
  var d = function() {};

  b = 3;

}

foo(1);

在进入实行高低文后,这时刻的 AO 是:

AO = {
    arguments: {
        0: 1,
        length: 1
    },
    a: 1,
    b: undefined,
    c: reference to function c(){},
    d: undefined
}

代码实行

在代码实行阶段,会递次实行代码,依据代码,修正变量对象的值

照样上面的例子,当代码实行完后,这时刻的 AO 是:

AO = {
    arguments: {
        0: 1,
        length: 1
    },
    a: 1,
    b: 3,
    c: reference to function c(){},
    d: reference to FunctionExpression "d"
}

到这里变量对象的建立历程就引见完了,让我们简约的总结我们上述所说:

  1. 全局高低文的变量对象初始化是全局对象

  2. 函数高低文的变量对象初始化只包含 Arguments 对象

  3. 在进入实行高低文时会给变量对象增加形参、函数声明、变量声明等初始的属性值

  4. 在代码实行阶段,会再次修正变量对象的属性值

思考题

末了让我们看几个例子:

1.第一题

function foo() {
    console.log(a);
    a = 1;
}

foo(); // ???

function bar() {
    a = 1;
    console.log(a);
}
bar(); // ???

第一段会报错:Uncaught ReferenceError: a is not defined

第二段会打印:1

这是由于函数中的 “a” 并没有经由过程 var 关键字声明,一切不会被存放在 AO 中。

第一段实行 console 的时刻, AO 的值是:

AO = {
    arguments: {
        length: 0
    }
}

没有 a 的值,然后就会到全局去找,全局也没有,所以会报错。

当第二段实行 console 的时刻,全局对象已被赋予了 a 属性,这时刻就能够从全局找到 a 的值,所以会打印 1。

2.第二题

console.log(foo);

function foo(){
    console.log("foo");
}

var foo = 1;

会打印函数,而不是 undefined 。

这是由于在进入实行高低文时,首先会处置惩罚函数声明,其次会处置惩罚变量声明,假如假如变量称号跟已声明的形式参数或函数雷同,则变量声明不会滋扰已存在的这类属性。

下一篇文章

《JavaScript深切之作用域链》

本文相干链接

《JavaScript深切之实行高低文栈》

深切系列

JavaScript深切系列目次地点:https://github.com/mqyqingfeng/Blog

JavaScript深切系列估计写十五篇摆布,旨在帮人人捋顺JavaScript底层学问,重点解说如原型、作用域、实行高低文、变量对象、this、闭包、按值通报、call、apply、bind、new、继续等难点观点。

假如有毛病或许不严谨的处所,请务必赋予斧正,非常谢谢。假如喜好或许有所启示,迎接star,对作者也是一种勉励。

    原文作者:冴羽
    原文地址: https://segmentfault.com/a/1190000009018898
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞