理解 JavaScript 作用域

上一篇文章中分析了 JS 中的数据类型和变量。这一篇文章将分析作用域,以及回答上一篇文章中变量提升的原因。

什么是作用域

作用域是一套规则,保存着变量,等待被引擎所查找。

var a = 1;
console.log(a);  // => 1
console.log(b);  // => ReferenceError

当打印 a 时,引擎就去作用域中查找 a,找到把结果返回。如果查找失败,那么就会报错。

词法作用域

JS 采用的词法作用域,也可以说是静态作用域。简单来说,词法作用域是由写代码时将变量写在哪里决定的。

先看一段代码:

var a = 1;function fn() {var a = 2;return a;
}fn();  // => 2

当执行函数 fn 时,会返回 2,而不是 1。

作用域查找

JS 引擎会进行两种查找,LHS 和 RHS。怎么理解?L 和 R 可以说代表左和右。什么的左和右?赋值操作的。

这里的赋值操作不一定出现 =,比如参数传递也是一个赋值操作。

当变量出现在赋值操作的左边时,引擎就会对这个变量进行 LHS 查找;当出现在右边时(这个还可以理解为取得变量的源值),就会进行 RHS 查找。

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

对于变量 a 来说,引擎会进行两次查找,1 次 LHS,1 次 RHS。

调用 foo(),并传入参数 2,这时存在着一个赋值操作即 a = 2,进行一次 LHS 查找。打印 a 时,需要获取 a 的源值,所以进行一次 RHS 查找。

如果查询失败呢?

对于 LHS 来说,给未声明的赋值就会查询失败。

a = 2;

但是我们知道,上面的代码在非严格模式下并不会报错,而变量 a 会被自动创建。

而对于 RHS 来说,直接使用未声明的变量就会报 ReferenceError。

console.log(a); // => ReferenceError

另外,RHS 虽然查询成功,但是却对查询结果进行非法操作,就会报 TypeError。

var foo = 1;
foo(); // => TypeError

作用域链

前面说,作用域是根据名称查找变量的一套规则。而在实际情况中,经常出现多个作用域嵌套的情况。

function foo(a) {console.log(a + b);
}
var b = 2;
foo(2); // => 4

当引擎对 b 进行 RHS 查找时,在当前作用域无法找到,引擎就会在外层作用域中查找,直到找到这个变量,或者直到抵达最外层作用域(全局作用域)为止。

LHS 查找也是如此。

把这样一层一层嵌套的作用域,叫做作用域链。

函数作用域

函数作用域是指,属于这个函数的全部变量都可以在这个函数的范围内使用及复用。

function foo() {var a = 1;
}console.log(a); // => ReferenceError

也就是说,函数外部将无法访问函数内部的变量。

但是这却是非常有用的。我们可以利用函数隐藏内部实现,使其外部无法访问、修改等。

立即执行函数表达式

利用函数作用域,可以将外部作用域无法访问的内容包装起来。但是,带来了额外的一个问题,函数名本身“污染”了所在的作用域。

这时,就提出了 IIFE(立即执行函数表达式)。

(function foo() {// ...
}());

即包装了内部函数,又避免了引入函数名。因为这个函数名无法被外部作用域所访问。

IIFE 的进阶用法是给其传入参数:

(function fn(global) {// ...
})(window);

这样的好处是可以缩短查询时的作用域链。

块作用域

ES6,通过 let 和 const 引入了块作用域。

if (true) {let a = 1;
}
console.log(a); // => ReferenceError

变量提升

上一篇文章中中提到了变量提升。

在 JS 中,var a = 1; 这行代码其实会被看成 var aa = 2,并在两个阶段去执行。

在编译阶段,执行声明操作;在执行阶段,执行赋值操作。

所有的变量声明都会被提升到作用域的顶部,这个过程叫做“提升”。

函数声明也会发生提升,并且函数声明会先于变量提升:

var foo = 1;
function foo () {}typeof foo; // => 'number'

注意,只有函数声明会被提升,而函数表达式不会被提升。

var foo = 1;
var foo = function () {}typeof foo; // => 'function'

小结

这篇文章梳理了 JavaScript 中作用域的基本知识。

接下来会介绍执行上下文和闭包这两个概念,它们与作用域息息相关。

关于

这是我的公众号,记录着我的前端博客,没事儿也分享一些电影、书籍。

欢迎一起交流学习。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/537426.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mysql行求和

SELECT 列1 列2 列3 …… 列N AS Total FROM 表 SELECT sum(列1 列2 列3 …… 列N) AS Total FROM 表 转载于:https://www.cnblogs.com/weilovehua/p/10024624.html

python安装后在哪里找_python安装后的目录在哪里

从官网下载python的安装包,安装过程中可选择装在C盘或D盘或者其他的磁盘。 如果忘记了安装在哪里,可以在命令行中使用以下命令 where python 会显示python的绝对路径 C:\Users\Administrator>where python C:\Users\Administrator\AppData\Local\Prog…

Axure原型设计导出到PDF文件

Axure 没有直接导出PDF文件的功能,可以通过Axure 的打印功能,选择PDF打印机,以间接的方式将原型设计导出到pdf文件里。 操作步骤 以Axure9为例 打印 Axure9---文件---打印 不要母版 预览 预览下效果,看下是否有不必要的内容 …

izoak028

离散数学 (自考) 【自考】计算机网络原理精讲(2018版)转载于:https://www.cnblogs.com/qianyindichang2/p/10025538.html

Word查找替换详细用法及通配符一览表

使用通配符 要查找“?”或者“*”&#xff0c;可输入“\?”和“\*”&#xff0c;\1\2\3依次匹配数对括号内容 查找(a)12(b) 替换\2XY\1 结果&#xff1a;bXYa ([.0-9]) [MG]B 匹配文件大小&#xff0c; 例1: 201 MB ,例2: 2.51 GB <(e*r)> 匹配“…

python pca降维_机器学习的降维打击

文章发布于公号【数智物语】 (ID&#xff1a;decision_engine)&#xff0c;关注公号不错过每一篇干货。来源 | SAMshare(id:SAMshare)作者 | samshare"本次主要讲解的内容就是特征降维&#xff0c;主要涉及PCA以及一些常见分析方法。"01Index一&#xff0c;PCA降维算…

什么样的项目是成功的?

项目成功的标准是什么&#xff1f; 项目范围控制住&#xff0c;成本没超标&#xff0c;质量达标&#xff0c;进度按计划&#xff0c;顺利验收。做到这些就是项目成功了吗&#xff1f; 答案显然是不一定&#xff01;&#xff01;! 有多少项目的成本、进度、目标都能够严格按照…

ng-notadd 0.10.1,基于 Angular7 和 material2 的中后台解决方案

更新内容修复 scss左侧导航栏美化修复导航栏 2px 间隔问题技术栈TypescriptAngularMaterial2rxjsGraphql相关链接项目地址DEMOng-notadd-mock-serverQuick startgit clone https://github.com/notadd/ng-notadd.gitcd ng-notaddnpm installnpm start# or use ng cling serve复制…

python需要什么包装_python学习之包装与授权

实现授权的关键点就是覆盖__getattr__()方法&#xff0c;在代码中包含一个对getattr()内建函数的调用。 特别调用getattr()以得到默认对象属性&#xff08;数据属性或者方法&#xff09;并返回它以便访问或调用。 特殊方法__getattr__()的工作方式是&#xff0c;当搜索一个属性…

参加技术培训前的辅导,选得对,学得好

最近几年&#xff0c;每年都会有人问我培训班的事情&#xff0c;我也有培训班经历&#xff0c;在软件行业工作了十多年&#xff0c;每次解答培训班的咨询我都很认真&#xff0c;也很高兴能帮到他人。 决定通过专栏的形式解答培训班常见问题&#xff0c;我把专栏取名“技术培训…

[算法]浅谈求n范围以内的质数(素数)

汗颜&#xff0c;数学符号表达今天才学会呀-_-# 下面是百度百科对质数的定义 质数&#xff08;prime number&#xff09;又称素数&#xff0c;有无限个。质数定义为在大于1的自然数中&#xff0c;除了1和它本身以外不再有其他因数。求质数的方法自然不少&#xff0c;但主要还是…

进入IT行业,要不要参加培训班?

IT行业介绍 考虑培训班无非是要入行,那IT行业好不好?IT行业当然好,看看培训班的数量就知道了。现在房产行业好赚钱,每个小区门口好几家中介门店,相同品牌的可能不止1家。不用去看网上的软文,也不用去问百度,看市场的反应,这是真实的反馈。培训班越来越多,课程越来越多…

python commands_Windows环境下使用python的commands.getstatusoutput

windows调用系统或其他脚本的&#xff0c;常用的是os.popen&#xff0c;次命令本身并不返回执行后的状态&#xff0c;无法用于后续的判断&#xff0c;故尝试Unix下的commands.getstatusoutput&#xff0c;发现在windows下并不能正常使用&#xff0c;如下&#xff1a; >>&…

Kubernetes在上汽集团云平台及AI方面的应用

2019独角兽企业重金招聘Python工程师标准>>> 帆一尚行成立于2015年&#xff0c;是上汽集团的全资子公司&#xff0c;建设有上海、南京、郑州&#xff08;在建&#xff09;三个数据中心&#xff0c;拥有超过4000台物理服务器&#xff0c;10PB的数据存储&#xff0c;总…

我的Java培训经历

此文讲述我的Java开发培训经历&#xff0c;来解答关心的培训费、培训节奏、就业等问题。 我在2010年参加达内Java培训&#xff0c;如今再回首那段时光&#xff0c;虽然辛苦&#xff0c;但很值得&#xff01;&#xff08;后悔参加培训班&#xff0c;大部分原因是冲动&#xff0…

python跨函数调用变量_对python中不同模块(函数、类、变量)的调用详解

首先&#xff0c;先介绍两种引入模块的方法。 法一&#xff1a;将整个文件引入 import 文件名 文件名.函数名( ) / 文件名.类名 通过这个方法可以运行另外一个文件里的函数 法二&#xff1a;只引入某个文件中一个类/函数/变量 需要从某个文件中引入多个函数或变量时&#xff0c…

软件培训技术选哪个?

要培训了,培训技术怎么选? 技术需慎重选 女怕嫁错郎,男怕入错行。后悔参加培训班,因为技术没选好的占比很高。 技术没选好会有什么影响? 近的影响是就业!远的影响是发展! 对于程序员来说,技术就是立身之本,需要慎重选择! 我在《要不要参加培训班?》文章中介绍…

django安装_技术大牛详解:Django框架之环境安装

黑马程序员视频库播妞微信号&#xff1a;boniu236传智播客旗下互联网资讯、学习资源免费分享平台虚拟环境安装:开发中问题&#xff1a;如何在同一台主机中&#xff0c;要开发多个不同的项目&#xff0c;而且需要用到同一个包的不同版本&#xff1f;尝试分析&#xff1a;在开发过…

安装 Alibaba Cloud Toolkit

IntelliJ IDEA版 JetBrains 插件市场下载 Eclipse 版 Eclipse 插件市场仓库下载 (推荐)URL 地址在线安装Maven 版 在 POM 文件中依赖 PyCharm、PhpStorm、RubyMine 和 WebStorm 版 公测中官网https://toolkit.aliyun.com 交流群&#xff08;钉钉&#xff09; 交流群&#xff08…

软件Java前端大数据培训机构怎么选?

先看这篇文章《要不要参加培训班》。 选技术就像选另一半,那选培训机构就是选另一半的家庭。另一半家庭好与不好,与婚后幸福生活息息相关。 选培训机构的几个维度: 1.成立时间 2.专业性 3.市场普及率 成立时间 成立久的不一定好,比如北大某鸟 成立不足3年的,不要选…