读汤姆大叔《JavaScript变量对象》笔记

一段简单的JavaScript代码思考

先看一段简单的代码,打印结果是??为什么why??

从上述打印结果不难看出,在打印基本变量num、函数表达式fn、函数声明fun时,就已经知道变量num、函数表达式fn值为undefined,函数声明fun值为function

但这样会存在一个疑问?js在代码执行之前打印时(变量、函数表达式、函数声明)就存在值,那么在打印(变量、函数表达式、函数声明)之前究竟发了什么??

【答案是变量对象、执行上下文代码的2个阶段】

(当然 打印在基本变量num、函数表达式fn、函数声明fun之后结果是不言自明的)

变量对象 variable object

如果变量与执行上下文相关,那么它自己应该知道它的数据存储在哪里和如何访问。这种机制被称作 变量对象(variable object).

一个变量对象完全有可能用正常的ECMAScript对象的形式来表现

VO = {};

VO就是执行上下文的属性(property)

activeExecutionContext = {
  VO: {
    // context data (var, FD, function arguments)
    arguments 函数的所有形式参数
    FunctionDeclaration 函数的声明
    variable Declaration 变量声明
  }
};

处理上下文代码的2个阶段

    1. 进入执行上下文(代码执行之前时)
    2. 执行代码

变量对象的修改变化与这两个阶段紧密相关

(个人理解标注 变量对象与执行上下文关系强调是代码执行前后过程中变量对象的变化)

为了更好的理解执行上下文分阶段处理(包括进入执行上下文、执行代码)我们来看下下面test函数执行上下文分阶段处理过程

var a = 10;
function test(x) {
  var b = 20;
};
test(30);

在上述代码第5行代码,调用test激活函数至第3行代码之前时(包括第3行) x 值为 30, b值undefined

其对应变量对象简单表示如下

// Variable object of the global context
VO(globalContext) = {
  a: 10,
  test:
};
// Variable object of the "test" function context
VO(test functionContext) = {
  x: 30,
  b:undefined
};  

执行test函数第3行代码后,函数test未弹出栈时,其对应的变量对象如下

// Variable object of the global context
VO(globalContext) = {
  a: 10,
  test:
};
// Variable object of the "test" function context
VO(test functionContext) = {
  x: 30,
  b: 20
};

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

(标注 这里激活可以理解调用函数, 现实中更形象例子向水中投入石子,一石激起千层浪)

VO(functionContext) === AO;    

且 执行上下文的属性(property)

activeExecutionContext = {
  VO: {}
}

于是上述函数执行上下文中变量对象表示代码修改成如下形式

// Variable object of the global context
activeExecutionContext = {
     // Variable object of the global context
     AO(globalContext) = {
       a: 10,
       test:
     };
     // Variable object of the "test" function context
     VO(test functionContext) = {
       x: 30,
       b: 20
     };
}

变量对象VO(variable object)就是执行上下文的属性(property)

变量对象中的变量在执行上下文中默认次序变量声明在顺序上跟在函数声明和形式参数声明之后

activeExecutionContext = {
  VO: {
    // context data (var, FD, function arguments)
    arguments 函数的所有形式参数(如果我们是在函数执行上下文中)
              (变量对象的一个属性,这个属性由一个形式参数的名称和值组成;如果没有对应传递实际参数,那么这个属性就由形式参数的名称和undefined值组成)
    FD 所有函数声明(FunctionDeclaration, FD)变量对象的一个属性,这个属性由一个函数对象(function-object)的名称和值组成;如果变量对象已经存在相同名称的属性,则完全替换这个属性。
    VD 所有变量声明(var, VariableDeclaration)
        变量对象的一个属性,这个属性由变量名称和undefined值组成;如果变量名称跟已经声明的形式参数或函数相同,则变量声明不会干扰已经存在的这类属性。
  }
};

代码体现

alert(x); // function
var x = 10;
alert(x); // 10
x = 20;
function x() {};
alert(x); // 20 

Tom大叔 深入理解JavaScript系列(12):变量对象(Variable Object

英文地址 : ECMA-262-3 in detail.Chapter 2.Variable object
中文地址 [JavaScript]ECMA-262-3 深入解析.第二章.变量对象

时间: 2024-10-30 14:23:13

读汤姆大叔《JavaScript变量对象》笔记的相关文章

汤姆大叔 javascript 系列 第20课 最后的5到javascript题目

博客链接:http://www.cnblogs.com/TomXu/archive/2012/02/10/2342098.html 原题: 大叔注:这些题目也是来自出这5个题目的人,当然如果你能答对4个及以上并且想拿高工资的话,请联系我. 找出数字数组中最大的元素(使用Match.max函数) 转化一个数字数组为function数组(每个function都弹出相应的数字) 给object数组进行排序(排序条件是每个元素对象的属性个数) 利用JavaScript打印出Fibonacci数(不使用全

我读汤姆大叔的深入理解js(二)

继续汤姆大叔的js之旅. 揭秘命名函数表达式 函数表达式和函数声明 汤姆大叔在博客中引用ECMA规范:函数声明必须带有标识符,函数表达式可以省略.对于我来说这些概念的东西真是不所适从.还是大叔的实例带劲.上实例如下: function foo(){};//鬼都知道是声明 var bar = function foo(){};//鬼也知道是表达式 new function bar(){}; (function() { function bar(){};//这也是声明 })();这些大家看看应该都很

读汤姆大叔《深入理解javascript系列》笔记一编写高质量代码

感觉大叔的博文真的是很不错 我打算严格要求自己 按照大叔说的,这样我就会更有规范,更有思想的去工作 去写代码(一入代码深似海) 1,尽可能的少用全局变量(污染全局空间,也会和别人的代码 发生冲突造成严重的后果) ---->命名空间模式或者是函数立即自动执行. 2.不管是不是在函数内始终用var来声明变量, 例如: function sum(x, y){ result = x + y; //这里创建了一个隐式全局变量 return result; } 3.避免出现隐式变量 例如在函数中不通过var

根本没有“JSON“对象这回事(读汤姆大叔博文记录)

1.字面量 (1)他们是固定的值,不是变量,让你从“字面上”理解脚本. (2)字符串字面量是由双引号("")或单引号('')包围起来的零个或多个字符串组成的. (3)对象字面量是由花括号({})括起来的零个或多个对象的属性(key-value对). 2.何时是JSON,何时不是JSON JSON是设计成描述数据交换格式的,他也有自己的语法,这是语法是JavaScript的子集. {"proporty":"value"}这样的声明可能是JavaSc

汤姆大叔JavaScript

全局变量的问题在于,你的JavaScript应用程序和web页面上的所有代码都共享了这些全局变量,他们住在同一个全局命名空间,所以当程序的两个不同部分定义同名但不同作用的全局变量的时候,命名冲突在所难免.JavaScript的两个特征,不自觉地创建出全局变量是出乎意料的容易.首先,你可以甚至不需要声明就可以使用变量:第二,JavaScript有隐含的全局概念,意味着你不声明的任何变量都会成为一个全局对象属性 function sum(x, y) { // 不推荐写法: 隐式全局变量 result

学习汤姆大叔的knockout教程笔记

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd"> <html> <head> <script type="text/javascript" src="http://ajax.googleapis.com/ajax/libs/jquery/1.4.0/jquery.m

学习汤姆大叔的knockout教程笔记(二)

模板绑定The template binding 目的 template绑定通过模板将数据render到页面.模板绑定对于构建嵌套结构的页面非常方便.默认情况, Knockout用的是流行的jquery.tmpl模板引擎.使用它的话,需要在安装页面下载和引用jquery.tmpl和jQuery框架.或者你也可以集成其它的模板引擎(虽然需要了解Knockout 内部知识才行). <!DOCTYPE html> <html lang="en"> <head&g

看了汤姆大叔的“你真懂JavaScript吗?”的一些感慨

看了汤姆大叔的“你真懂JavaScript吗?”,里面有5道题目,我都一一作了,然后在chrome的控制台里面运行了一遍,虽然只错了一道,但还是细细读了下答案,在此总结一下,看看是否对大家对这些JavaScript底层的原理都懂了. 题目一(所有全局变量都是window的属性.变量声明提前.变量赋值不会提前) if (!("a" in window)) { var a = 1; } alert(a); 因为在JavaScript在变量声明提前的特性,所以事实上上述代码相当于下面所示:

深入理解JavaScript系列 --汤姆大叔

深入理解JavaScript系列文章,包括了原创,翻译,转载,整理等各类型文章,如果对你有用,请推荐支持一把,给大叔写作的动力. 深入理解JavaScript系列(1):编写高质量JavaScript代码的基本要点 深入理解JavaScript系列(2):揭秘命名函数表达式 深入理解JavaScript系列(3):全面解析Module模式 深入理解JavaScript系列(4):立即调用的函数表达式 深入理解JavaScript系列(5):强大的原型和原型链 深入理解JavaScript系列(6