《你不知道的javascript(上)》作用域与闭包(一)

在翻阅《你不知道的javascript》这一套书的中上卷目录之后,发现书中针对闭包、对象、原型、语法、异步、回调等等既基础又重要的
javascript知识有着针对性的阐述,于是决定对这套书的中上卷进行学习。上卷和中卷各讲述了两大部分知识,分别是:作用域与闭包、
this和对象原型、类型和语法、异步和性能。本文是对作用域与闭包的学习总结。

在第一部分中,尽管分为六章来进行叙述,但我认为主要分为三大块知识,分别是编译原理、作用域以及闭包机制,依托于javascript引擎的编译原理规则,才能够形成作用域的理念,而依托于作用域的理念,才能够体现出闭包机制的作用。

对编译原理的掌握,需要了解的是:javascript编译的步骤,引擎、编译器、作用域三者之间的相互关系,以及查询机制。

1.javascript编译的步骤

  一般情况下,编程语言编译的步骤都是分词/词法分析、解析/语法分析、代码生成这样三步:

  • 分词/词法分析:将字符串分解成代码块(也叫词法单元)就是分词,如果是在按照某种规则下进行的分词,就是词法分析;
  • 解析/语法分析:将词法单元流转换成一个由元素逐级嵌套所组成的代表程序语法结构的树(抽象语法树,AST)。
    javascript中的作用域基本上就是在这一阶段形成,也叫词法作用域。
  • 代码生成:将AST转换成引擎可执行的代码的过程。

然而对于js引擎来说,编译过程要复杂一些:由于js引擎的大部分编译工作发生在代码执行前很短时间内(非常非常短),因此在编译的时候,js引擎还需要做对代码的运行性能优化这一项工作(此处是重点,因为性能优化是发生在这一个阶段,所以在编写代码的时候,需要注意使代码不干预这一阶段的工作,在这一部分的后面章节中会有提到)。

2.引擎、编译器、作用域三者之间的关系

  首先需要明确的一点是,编译器也是引擎的一部分,将其脱离出来单独作为主体,是为了更好的阐述这三方面的关系。简单来说,由编译器通过分析词法来生成作用域中的变量、标识符等这样一些东西;而引擎在执行代码的过程中,会在作用域中查找需要用到的标识符;作用域可以理解为代码执行及标识符储存的范围。
  简单以下声明为例来具体分析三者的作用:

var a=2;

在对这个声明执行过程中,编译器会起到两个作用:
a.在作用域中查询a,如果a不存在则在作用域中声明a这个变量;
b.为引擎生成执行赋值这一动作的代码。

  而引擎也起到两个作用:
a.在作用域中查询a;
b.对a进行赋值。

3.查询机制
  在上面一部分中,提到了查询,由此,需要了解引擎的查询机制。查询机制分为两种:LHS查询和RHS查询。在书中对两种查询机制的原意进行了较为全面的阐述,总结下来,两种查询机制的关键点在:

  • LHS查询:找到变量并为了给变量赋值,可理解为“赋值操作的目标是谁”;

  • RHS查询:查找变量,确认有无变量,可理解为“谁是赋值操作的源头”;


      同样,简单以以下一段代码为例,进行具体分析:

      function foo(a) {
        console.log(a);
      }
      foo(2);
    

1.首先,引擎对foo这一标识符进行RHS查询,看是否存在,发现编译器已经声明的foo,代表一个函数;

2.接着,引擎在执行函数阶段,对a执行了LHS查询,并将2这个值赋给它;

3.接着,引擎对console进行了RHS查询,看是否存在,发现console及其子函数log()都是内置的方法;

4.最后,引擎在log(a)中又看到了a,于是对a进行了一次RHS查询,发现a存在,并且在第二步的LHS查询中已经被赋值为2,于是将这个2传递进log(a)中的a。

两种查询都存在异常的情况,其异常也存在区别:

  • 对RHS查询来说,如果在任何相关作用域(此处需要了解作用域链的原理)中都找不到变量,引擎就会抛出ReferenceError异常;
  • 对LHS查询来说,如果找不到变量,会存在两种情况:非严格模式下,如果查不到变量,全局作用域会创建一个该变量,并返还引擎(此处需要注意,一般来说,js的作用域都是词法作用域,静态的,但在这种情况下,作用域却发生了改变,被语法进行了欺骗);在严格模式下(ES5引进了严格模式),引擎同样会抛出ReferenceError异常,而作用域不会出现创建变量的动作。
    此外,对查询结果的非法或不合理操作(存在于LHS查询之后),引擎会抛出TypeError异常。

以上,就是编译原理方面的知识。

Small Star's Blog|小星的博客

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,029评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,395评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,570评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,535评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,650评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,850评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,006评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,747评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,207评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,536评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,683评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,342评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,964评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,772评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,004评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,401评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,566评论 2 349

推荐阅读更多精彩内容