资讯专栏INFORMATION COLUMN

5分钟深入了解js变量提升

tinysun1234 / 1410人阅读

摘要:对于大多数开发者来说,变量提升可以说是一个非常常见的问题,但是可能很多人对其不是特别的了解。如果说拥有和一样的变量提升效果的话,那么应该是输出。而和它们的变量提升的效果是一样的,也都存在着临死性死区的概念。

对于大多数js开发者来说,变量提升可以说是一个非常常见的问题,但是可能很多人对其不是特别的了解。所以在此,我想来讲一讲。

先从一个简单的例子来入门:

</>复制代码

  1. a = 2;
  2. var a;
  3. console.log(a);

你觉得以上的代码会输出什么?是输出undefined吗?如果是按照程序的自上而下执行的话,那么这一段代码确实是输出undefined,因为var a语句把a = 2的语句给覆盖了。然而,javascript并不是严格的自上而下执行的语言

这一段代码的输出结果是2,是不是感到很意外?为什么会这样呢?这个问题的关键就在于变量提升(hoisting)。它会将当前作用域的所有变量的声明提升到程序的顶部,因此上面的代码其实等价于以下代码。这样是不是就很简单明了了。

</>复制代码

  1. var a;
  2. a = 2;
  3. console.log(a);

那么接下来,我们再来看这个例子。

</>复制代码

  1. console.log(a);
  2. var a = 2;

你觉得以上的代码会输出什么?是直接报ReferenceError吗?还是输出2呢?

其实以上代码会输出undefined。为什么呢?我们之前说过,js会将变量的声明提升到顶部,可是赋值语句并不会提升。所以对于js来说,其实var a = 2是分为两步的:

var a;

a = 2;

而js只会将第一步提升到顶部,所以上面的语句等价于:

</>复制代码

  1. var a;
  2. console.log(a);
  3. a = 2;
为什么有变量提升

那么为什么会出现变量提升这个现象呢?

js和其他语言一样,都要经历编译和执行阶段。而js在编译阶段的时候,会搜集所有的变量声明并且提前声明变量,而不会改变其他语句的顺序,因此,在编译阶段的时候,第一步就已经执行了,而第二步则是在执行阶段执行到该语句的时候才执行。

变量声明

js的变量声明其实大体上可以分为三种:var声明、let与const声明和函数声明。

函数声明与其他声明一起出现的时候,就可能会引起一些困扰。我们来看下面的例子。

</>复制代码

  1. foo();
  2. function foo() {
  3. console.log("foo");
  4. }
  5. var foo = 2;

你觉得上面会输出什么?TypeError吗?还是输出foo呢?想一想再接着往下看。

当当当当,其实最后输出的结果是foo。这就引出了我们的问题了,当函数声明与其他声明一起出现的时候,是以谁为准呢?答案就是,函数声明高于一切,毕竟函数是js的第一公民。

那么,下面的例子呢?

</>复制代码

  1. foo();
  2. function foo() {
  3. console.log("1");
  4. }
  5. function foo() {
  6. console.log("2");
  7. }

当出现多个函数声明,那怎么办呢?

上面这个代码输出结果为2。因为有多个函数声明的时候,是由最后面的函数声明来替代前面的。

想必经历了以上的例子,你应该已经对变量声明已经有一定的了解了。那么我再来出一道题目来测试下。

</>复制代码

  1. foo();
  2. var foo = function() {
  3. console.log("foo");
  4. }

这道题目是不是非常简单啊?这道题和上面的第二道例子其实是一样的。var foo = function() {}这种格式我们叫做函数表达式。

它其实也是分为两部分,一部分是var foo,而一部分是foo = function() {},参照例2,我们可以知道,这道题的结果应该是报了TypeError(因为foo声明但未赋值,因此foo是undefined)。

上面我们提到了var声明,函数声明,那么接下来我们来讲讲let和const声明。

let和const其实也是有函数提升的概念,不过它们会比较特殊,与var不一样,它们存在一个临死性死区的概念。我们可以通过一个例子来体现这一点。

</>复制代码

  1. var a = 2;
  2. function test() {
  3. console.log(a);
  4. let a = 5;
  5. }
  6. test();

你觉得上面的代码会输出什么呢?是输出2,还是undefined呢?

如果说,let没有变量提升的效果的话,那么应该是输出2。如果说let拥有和var一样的变量提升效果的话,那么应该是输出undefined。然而,其实上面的代码是会报错的。会报"ReferenceError: a is not defined"错误。那么为什么呢?

这其实就是我说的,let虽然具有变量提升的功能,但是它又与var不一样,它具有一个临死性死区的概念。

临死性死区其实就是说,a我已经声明了,可是在没有到它赋值的时候,你都不能使用这个变量,不然就会报错。所以该当前作用域开始,一直到let a = 5为止这整一块,都是a变量的临死性死区,你不能使用它。

而const和let它们的变量提升的效果是一样的,也都存在着临死性死区的概念。

总结

那么接下来我们来总结一下。

js会将变量的声明提升到js顶部执行,因此对于这种语句:var a = 2;其实上js会将其分为var a;和a = 2;两部分,并且将var a这一步提升到顶部执行。

变量提升的本质其实是由于js引擎在编译的时候,就将所有的变量声明了,因此在执行的时候,所有的变量都已经完成声明。

当有多个同名变量声明的时候,函数声明会覆盖其他的声明。如果有多个函数声明,则是由最后的一个函数声明覆盖之前所有的声明。

let和const都具有变量提升的效果,但是它们都具有临死性死区,从作用域开始部门,一直到变量的声明语句这整一块,你都不能使用该变量。

以上,就是本文的内容。觉得有用的话麻烦点个赞,嘻嘻。

本文地址在->本人博客地址, 欢迎给个 start 或 follow

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/98942.html

相关文章

  • JavasScript重难点知识

    摘要:忍者级别的函数操作对于什么是匿名函数,这里就不做过多介绍了。我们需要知道的是,对于而言,匿名函数是一个很重要且具有逻辑性的特性。通常,匿名函数的使用情况是创建一个供以后使用的函数。 JS 中的递归 递归, 递归基础, 斐波那契数列, 使用递归方式深拷贝, 自定义事件添加 这一次,彻底弄懂 JavaScript 执行机制 本文的目的就是要保证你彻底弄懂javascript的执行机制,如果...

    forsigner 评论0 收藏0
  • 你需要知道面试中的10个JavaScript概念

    摘要:自我学习目前有成千上万的年轻人在学习和开发,希望获得一份工作。知道的绑定规则。知道和原型属性是什么以及它们的作用。高阶函数了解函数是中的一级对象,这意味着什么知道从另一个函数返回函数是完全合法的。了解闭包和高阶函数允许我们使用的情况。 翻译原文出处:10 JavaScript concepts you need to know for interviews 之前不是闹得沸沸扬扬的大漠穷...

    YacaToy 评论0 收藏0
  • 体验jaivascript之美第二课 全局对象、外部环境、声明提升

    摘要:全局对象在程序任何地方都能访问到的对象,在浏览器中是变量指向在中是什么外部环境暂时不讲。函数声明直接提前。执行过程,函数执行和销毁过程。变量预解析,赋值外部环境之间的关系。如果做不对,继续找,直到合格。 当程序在浏览器中运行的时候,js引擎会产生三个东西。 showImg(https://segmentfault.com/img/bVMjv4?w=890&h=422);1.全局对象 ...

    icyfire 评论0 收藏0
  • 从国企到阿里的面试经历(一)

    摘要:月初,一个和我关系不错的阿里朋友,在内网看到一个开发专家的岗位,挺适合我,便问我有没有兴趣。正式开始电面正式开始,下文中阿里面试官简称。晚上点左右,接到阿里影业的来电,我赶紧带好耳机接起电话。 读完本文需要 3 分钟。 首先,跟大家说声抱歉,年后实在是太忙,带娃、录视频、换工作,大堆事情挤在一起,一不小心就断更了 4 个月,实在是对不起一直以来支持我的朋友们。 从今天开始,我尽量坚持...

    justjavac 评论0 收藏0

发表评论

0条评论

tinysun1234

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<