JavaScript变量对象其二:VO在不同的执行上下文中

不同执行上下文中的变量对象

对于所有类型的执行上下文来说,变量对象的一些操作(如变量初始化)和行为都是共通的。从这个角度来看,把变量对象作为抽象的基本事物来理解更为容易。同样在函数上下文中也定义和变量对象相关的额外内容。

1抽象变量对象VO (变量初始化过程的一般行为)
2 
3  ╠══> 全局上下文变量对象GlobalContextVO
4  ║        (VO === this === global)
5 
6  ╚══> 函数上下文变量对象FunctionContextVO
7           (VO === AO, 并且添加了<arguments>和<formal parameters>)

我们来详细看一下:

全局上下文中的变量对象

首先,我们要给全局对象一个明确的定义:

全局对象(Global object) 是在进入任何执行上下文之前就已经创建了的对象。这个对象只存在一份,它的属性在程序中任何地方都可以访问,全局对象的生命周期终止于程序退出那一刻。

全局对象初始创建阶段将Math、String、Date、parseInt作为自身属性,等属性初始化,同样也可以有额外创建的其它对象作为属性(其可以指向到全局对象自身)。例如,在DOM中,全局对象的window属性就可以引用全局对象自身(当然,并不是所有的具体实现都是这样):

1global = {
2  Math: <...>,
3  String: <...>
4  ...
5  ...
6  window: global //引用自身
7};

当访问全局对象的属性时通常会忽略掉前缀,这是因为全局对象是不能通过名称直接访问的。不过我们依然可以通过全局上下文的this来访问全局对象,同样也可以递归引用自身。例如,DOM中的window。综上所述,代码可以简写为:

1String(10); // 就是global.String(10);
2
3// 带有前缀
4window.a = 10; // === global.window.a = 10 === global.a = 10;
5this.b = 20; // global.b = 20;

因此,回到全局上下文中的变量对象——在这里,变量对象就是全局对象自己:

1VO(globalContext) === global;

非常有必要要理解上述结论,基于这个原理,在全局上下文中声明的对应,我们才可以间接通过全局对象的属性来访问它(例如,事先不知道变量名称)。

1var a = new String('test');
2
3alert(a); // 直接访问,在VO(globalContext)里找到:"test"
4
5alert(window['a']); // 间接通过global访问:global === VO(globalContext): "test"
6alert(a === this.a); // true
7
8var aKey = 'a';
9alert(window[aKey]); // 间接通过动态属性名称访问:"test"
函数上下文中的变量对象

在函数执行上下文中,VO是不能直接访问的,此时由活动对象(activation object,缩写为AO)扮演VO的角色。

1VO(functionContext) === AO;

活动对象是在进入函数上下文时刻被创建的,它通过函数的arguments属性初始化。arguments属性的值是Arguments对象:

1AO = {
2  arguments: <ArgO>
3};

Arguments对象是活动对象的一个属性,它包括如下属性:

  1. callee — 指向当前函数的引用
  2. length — 真正传递的参数个数
  3. properties-indexes (字符串类型的整数) 属性的值就是函数的参数值(按参数列表从左到右排列)。 properties-indexes内部元素的个数等于arguments.length. properties-indexes 的值和实际传递进来的参数之间是共享的。

例如:

01function foo(x, y, z) {
02
03  // 声明的函数参数数量arguments (x, y, z)
04  alert(foo.length); // 3
05
06  // 真正传进来的参数个数(only x, y)
07  alert(arguments.length); // 2
08
09  // 参数的callee是函数自身
10  alert(arguments.callee === foo); // true
11
12  // 参数共享
13  alert(x === arguments[0]); // true
14  alert(x); // 10
15
16  arguments[0] = 20;
17  alert(x); // 20
18
19  x = 30;
20  alert(arguments[0]); // 30
21
22  // 不过,没有传进来的参数z,和参数的第3个索引值是不共享的
23  z = 40;
24  alert(arguments[2]); // undefined
25
26  arguments[2] = 50;
27  alert(z); // 40
28}
29
30foo(10, 20);

这个例子的代码,在当前版本的Google Chrome浏览器里有一个bug  — 即使没有传递参数z,z和arguments[2]仍然是共享的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值