JavaScript中作用域相关的那些点

本文为《你不知道的JavaScript(上卷)》中关于作用域相关的知识点的总结。

作用域

赋值操作

变量的赋值操作实际上有两个动作,首先编译器会在当前作用域中声明一个变量(如果之前没有声明过),然后在运行时引擎会在作用域中查找该变量,如果能够找到就对它进行赋值。

LHS以及RHS

在运行时引擎会在作用域中查找该变量

引擎对变量所做的查找分为LHS查询以及RHS查询LR分别代表一个赋值操作的左侧以及右侧。

讲的稍微精确一点:RHS查询与简单地查找某个变量的值别无二致,而LHS则是试图查找到变量的容器本身,从而对其进行赋值。

RHS可以理解成retrieve his source value(取到其源值),这意味着“得到某某的源值”。

深入一点:

console.log(a);

上诉代码对于a的引用就是一个RHS引用,即查找然后取到a的值。

相比之下,

a = 2;

这里的a就是一个RHS引用。

我们可以简单的记忆:

当变量出现在赋值操作的左侧时进行LHS查询,出现在赋值操作的右侧时进行RHS查询.

注意:作用域查找会在找到第一个匹配的标识符时停止

作用域嵌套

作用域是根据名称查找变量的一套规则

作用域嵌套的定义如下:

当一个块或者函数嵌套在另一个块或函数中时,就发生了作用域的嵌套。

理解作用域嵌套这一机制,我们就可以理解变量查找的顺序:

  1. 在当前作用域查找变量。如果没有,则进行下一步
  2. 判断是否是全局作用域。如果是,则停止查找过程;如果不是,则进行下一步
  3. 进入当前作用域的外层作用域,并进行第一步

形象一点,我们可以把作用域查找想象成在大楼中找人。

第一层代表当前作用域,大楼的顶层代表全局作用域。

首先在当前楼层查找,如果没有找到,则上一楼进行查找,一直到找到这个人或者找完整个大楼依然没有找到为止。

异常报错的种类

如果能将LHS以及RHS进行很好的区分,那我们就能够很好的理解浏览器所抛出的各种异常。

下举几种特别常见的报错:

  • ReferenceError:
    1. RHS查询变量未找到值
    2. 严格模式LHS查询失败
  • TypeError:
    1. RHS找到该变量值,但尝试对这个变量的值进行不合理的操作(例如,引用null或者undefined类型的值中的属性)

词法作用域

词法作用域完全由写代码期间函数所声明的位置来定义

欺骗词法作用域

注意:欺骗词法作用域会导致性能下降

eval

eval() 是一个危险的函数, 他执行的代码拥有着执行者的权利。如果你运行eval()伴随着字符串,那么你的代码可能被恶意方(不怀好意的人)影响, 通过在使用方的机器上使用恶意代码,可能让你失去在网页或者扩展程序上的权限。更重要的是,第三方代码可以看到作用域在某一个eval()被调用的时候,这有可能导致一些不同方式的攻击。相似的Function就是不容易被攻击的。

with

根据你所传递给它的对象凭空创建了一个全新的词法作用域

性能问题

欺骗词法作用域会导致性能下降,其原因在于编译阶段的性能优化不起作用

JavaScript引擎会在即时编译阶段(during the compilation phase)进行数项的性能优化。其中的某些优化依赖于能够根据代码的词法进行静态分析,并预先确定所有变量和函数的定义位置,才能在执行的过程中快速找到标识符。

但是,编译到含有evalwith的代码时,编译器无法知道eval或者with会接受什么代码,自然无法做代码优化。

函数作用域以及块作用域

函数作用域:属于这个函数的全部变量都可以在整个函数的范围内使用及复用(事实上在嵌套的作用域中也可以使用)。

隐藏组件内部实现

开发者最主要是利用函数作用域实现隐藏组件或者API的内部实现,最小限度的暴露必要内容。

比如对于一些组件的开发,大家习惯于利用立即执行函数(function() {})()进行内部实现的封装。

规避冲突

利用函数作用域将变量保持在私有、无冲突的作用域中,这样可以有效规避掉所有的冲突。

举个例子,underscore这个库里面有跟原生js一样的方法map,那怎么区分这两个方法呢?通过将map当做一个属性挂载在underscore上面,这样可以避免两者的冲突。

立即执行函数表达式

形式如下:

  1. (function() {...})()
  2. (function() {...})()

上面两种形式没有区别,可依个人兴趣随意使用。

立即执行函数表达式的一种进阶用法就是把它们当做函数调用并传递参数进去。

各种类库常见的用法是:

(function(global) {
    ...
})(window)

块作用域

块作用域目前在ES6中有如下体现:

  1. let
  2. const
  3. with:用with从对象创建出的作用域仅在with声明而非外部作用域中有效。
  4. try/catchcatch分句会创建一个块作用域,其中声明的变量仅在catch内部有效。

例如:

for (let i; i < 4; i ++) {
    ...
}

console.log(i) // Uncaught ReferenceError: i is not defined
try {
    undefined();
} catch (err) {
    console.log(err);
}

console.log(err); // Uncaught ReferenceError: err is not defined

作用域闭包

知乎上面有关于闭包的问题:什么是闭包?

其中寸志老师的解释我认为是比较好的。

对于闭包,《你不知道的JavaScript(上卷)》这本书的解释是:

当函数可以记住并访问所在的词法作用域时,就产生了闭包。

我们实际上来理解闭包时,需要特别注意是两个点:函数作用域

简单的来说,就是函数以及作用域的结合,注意,作用域必须是封闭的,其主要的表现形式就是函数中返回一个函数。

闭包在类库、组件封装中有太多的示例了,本文就不拓展了。

块作用域与闭包的结合

首先看一个单纯的闭包的代码:

for (var i = 0; i <= 5; i++) {
    (function() {
       var j = i;
       setTimeout(function timer(){
           console.log(j);
       }, j * 1000) 
    })()
}

这段代码就是在每次循环的时候创建一个新的封闭作用域,保存当次循环的i值。

再看一下下面的代码:

for (let i = 0; i <= 5; i++) {
    setTimeout(function timer(){
        console.log(i);
    }, i*1000)
}

利用let创建块作用域,当块作用域与闭包结合之后,我们可以减少创建新的封闭作用域这一操作(var j = i);

that's cool!

动态词法作用域

动态作用域链是基于调用栈的,而不是代码中的作用域嵌套。

对于JavaScript,不存在动态作用域。如果一定要找一个点与动态词法作用域扯上关系的话,那就是this值了。this值打算在下一篇文章中详解。

变量提升

举个最简单的例子:

alert(a); // undefined
var a = 12;

有同样作用的是函数声明function,例如:

alert(func); // function func(){}
function func() {};

但是函数表达式不会提升:

foo(); // TypeError

var foo = function bar() {
    ...
}

注意:仅有var和函数声明function才可以变量提升。

函数声明与函数表达式的区别:

区别函数声明和函数表达式最简单的方法是看function关键字出现在声明中的位置(不仅仅是一行代码,而是整个声明中的位置)。如果function是声明中的第一个词,那么就是一个函数声明,否则就是一个函数表达式。

ES6中新增的let以及const关键字不可以进行变量提升,我们可以尝试一下:

// 1. let
alert(a); // Uncaught ReferenceError: a is not defined
let a = 'abc';

// 2. const
alert(b); // Uncaught ReferenceError: b is not defined
const b = 123;

函数优先

先来看下面的代码:

foo(); // 1
var foo;

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

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

上面的例子说明:
函数会被首先提升,然后才是变量

上面的代码实际等于:

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

foo(); // 1

var foo;

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

模块

模块这一利器,在以前封装插件用的非常多,示例如下:

var foo = (function CoolModule() {
    var something = "cool";
    var another = [1, 2, 3];
    
    function doSomething() {
        console.log(something)
    }
    
    function doAnother() {
        console.log(another.join("!"));
    }
    
    return  {
        doSomething: doSomething,
        doAnother: doAnother
    }
})()

模块模式必备条件如下:

  1. 必须有外部的封闭函数,该函数必须至少被调用一次(每次调用都会创建一个新的莫模块实例)。
  2. 封闭函数必须返回至少一个内部函数,这样内部函数才能在私有作用域中形成闭包,并且可以访问或者修改私有的状态。

当然,说到模块,我们不得不提到CMDAMDES6 module等模块机制了。

知乎上有提到AMD 和 CMD 的区别有哪些?

我这里简单提一下两者的区别:

  • AMD:
    • early executing(提前执行)
    • 推荐依赖前置
    • 示例:requireJs
  • CMD:
    • as lazy as possible(延迟执行)
    • 推荐依赖就近
    • 示例:seaJs

继续聊一下ES6的模块机制(importexport)。

import可以将一个模块中的一个或多个API导入到当前的作用域中,并分别绑定在一个变量上。

export会将当前模块的一个标识符(变量、函数)导出为公共API。

Github有很多基于es6实现的代码功能,请自行查阅。

好了,作用域相关的点整理完了,我将其中主要分成三部分:

  1. 作用域
  2. 提升
  3. 模块

如果有遗漏,欢迎指正~

PS:本文首发于segmenfault本人专栏

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

推荐阅读更多精彩内容