美文网首页
JS变量提升

JS变量提升

作者: _BuzzLy | 来源:发表于2020-04-27 18:59 被阅读0次

什么是变量提升

首先我们要知道js执行前有一个“预编译”过程,
预编译主要有两个任务:

  1. 声明所有var变量(初始为undefined)。
  2. 解析定义式函数语句。

==也就是说变量的提升实在js的预编译阶段完成的。==

变量提升的概念:函数和变量的声明会被js的解释器放到最上面。
在ES6之前,JavaScript没有块级作用域(一对花括号{}即为一个块级作用域),只有全局作用域和函数作用域。

变量提升

  • 栗子1
    function fn1() {
      a = 1;
      console.log(a); 
      console.log(window.a); 
      var a = 5;
      console.log(a); 
    }
    fn1();

输出结果依次为:1 undefined 5,因为代码在解析的时候相当于

      var a;//函数的声明会被解释器放到头部,预先声明,但没有赋值,所以在此刻a是undefined
      a = 1;
      console.log(a); 
      console.log(window.a); //此处打印的是全局变量的a,但是并没有声明全局变量a
      a = 5;
      console.log(a); 

  • 栗子2
    function fn1() {
      console.log(a); // a is not defined
      a = 5;
      console.log(a); // 如果不考虑报错,输出5
    }
    fn1();

变量提升一定是要有变量声明的过程,如var xx=5。像上面栗子中如果不声明直接对a赋值,那么a将变成一个全局对象。不存在变量提升。所以第一个console会报错。


  • 栗子3
    //全局作用域下
    a = 5;
    //全局作用域下
    var a = 5;

都在全局作用域下执行两端代码有何区别?

  • a=1相当于window.a=5。为全局window对象添加了一个属性a值为5
  • var a=5相当于在全局作用域中声明了变量a,在整个作用域中都有效
  • 后者比前者多了一个声明的行为
  • 前者没有变量提升过程,提前访问会报错,后者有变量提升

  • 栗子4
function fn2(){
    console.log(a);//undefined
    if(true){
        var a = 10;
    }
    console.log(a);//10
}

这个栗子中,a变量的声明同样被提升了。所以说==if是没有作用域的==。只有函数有作用域。(for循环等也是一样的)

函数的提升

js中我们定义函数有两种方式:

  • 函数表达式 var fn = function fn(){}
  • 函数声明方式,也叫定义式 function fn(){}

当函数是通过函数声明的方式定义时,就会有函数提升,
特别注意的是:

  • ==变量提升中,变量赋值并没有被提升,只是声明被提升了。==
  • ==但是,函数提升有点不一样,函数体也会一同被提升。==
test();

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

所以会出现上面的情况,先执行test函数,然后声明,但是实际上函数确实被执行了,这就是与变量提升不同的点,==函数提升不止声明提升,函数体也会一同被提升==。

  • 特别注意通过函数表达式创建的函数只会将声明提升,函数体不会提升
fn3();//fn3 is not a function

var fn3 = function(){}

再看两个示例:
1:

function bar() {
    console.log('bar1')
}

var bar = function () {
    console.log('bar2')
}

bar()

2:

var bar = function () {
    console.log('bar2')
}

function bar() {
    console.log('bar1')
}

bar()

结果全部输出 bar2,在预编译阶段,变量 bar 进行声明,但不赋值。函数bar进行创建并提升。运行阶段bar被赋值。

补充——es6中的变量提升

es6 新增块级作用域,let 和 const 声明的变量会产生块级作用域。
并且 let 和 const 声明的变量不会做变量提升。所以就会出现暂时性死区。

foo(10)
function foo(num) {
    // 暂时性死区
    console.log(foo)
    // 暂时性死区
    let foo = num
}

此时,浏览器报错foo is not defined。这就是因为 let 和 const 声明的变量不会做变量提升,所以变量的声明和赋值都是在console.log 之后,所以访问一个不存在,没有声明的变量时,浏览器自然会报错。

相关文章

  • JS中的提升

    JS中包含两种提升,变量提升和函数提升。 变量提升 变量提升只能是var或者function声明的变量或者函数,l...

  • 浏览器学习笔记-JS执行

    变量提升 变量提升原理浏览器对js是先编译后执行,在编译过程中,js中的变量声明会被提升到代码段落前面。函数声明和...

  • JavaScript 大纲

    js 语法基础 JavaScript 介绍 js 输出 注释 变量,变量声明提升,全局变量,常量 7 大数据类型 ...

  • Javascript 变量执行过程 和 数组 & 对象的区别

    变量提升 JS执行过程1. 先提升(先提升函数,再提升变量,如果名字一样,变量提升会覆盖函数提升)2. 再执行,如...

  • 变量提升和函数提升

    JS不像C语言,C语言是先声明后使用,否则会报错。但JS中,有变量提升现象,可以先使用后声明。 JS存在变量提升,...

  • 前端经典面试题合集(一)

    1.谈谈变量提升 考察点:js基础知识,js执行机制,变量的提升答:执行js代码时,会生成执行环境,在函数中的代码...

  • 变量声明提升

    变量声明提升是JS中一个基础的问题,同时也是对JS词法作用域认识的一个提升。在JS面试题中,关于变量声明提升的问题...

  • JS变量提升

    废话不多说,直接看以下例子,代码地址: 函数声明和函数表达式 很明显,这个大家都知道这个答案,就是很常见的作用域提...

  • js变量提升

    JavaScript的函数在查找变量时从自身函数定义开始,从“内”向“外”查找。如果内部函数定义了与外部函数重名的...

  • js变量提升

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

网友评论

      本文标题:JS变量提升

      本文链接:https://www.haomeiwen.com/subject/fvvuwhtx.html