美文网首页
变量提升和函数提升的问题

变量提升和函数提升的问题

作者: tency小七 | 来源:发表于2019-10-27 21:00 被阅读0次

关于变量提升,大家可能都听的耳朵都起茧了。但是最近的一次讨论 ,又让我有了新的认识。

先看看题:请说出1,2对应序号的结果
alert(a)    1
a();   2
function a(){
    alert(10)
} 
var a=3;

很明显由于变量提升,在执行alert(a)之前,function a和变量a已经被初始化了,所以alert(a)其实是有值的,而不是is not defined。
让我们来看看答案:

  1. f a(){alert(10)}
    关于这个答案,当时产生的第一个疑问就是,同样命名,为什么提升之后a反而是function a(){alert(10)},明明是函数a先提升的,变量a后提升的,为什么打印出来的不是undefined?曾经在网上看到这么一句话,函数优先级要高于变量优先级,相同命名,变量会被覆盖。不知道背后的逻辑究竟是怎么样的。看了https://zhuanlan.zhihu.com/p/28140450这篇文章之后恍然大悟。
    对于函数声明来讲,像如下代码

     console.log(a)
     function a(){
         console.log('helloworld')
     }
    

    创建,初始化,赋值的过程如下:

    • 找到所有用 function 声明的变量,在环境中「创建」这些变量。
    • 将这些变量「初始化」并「赋值」为 function(){ console.log(2) }。
    • 开始执行代码 fn2()
      所以,对于函数声明来讲,在代码执行之前,JS引擎就会对其进行创建,初始化,赋值
      函数表达式就只有创建和初始化过程。

    对于变量表达式/变量声明来讲,像如下代码

       console.log(a)
       var a = 2;
    
    • 找到所有用 var 声明的变量,在环境中「创建」这些变量。「创建」这些变量(即a)。
    • 将这些变量「初始化」为 undefined。所以上面打印出来的结果应该是undefined。
    • 执行代码
    • a = 2 将 a 变量「赋值」为 2
    • 也就是说 var 声明会在代码执行之前就将「创建变量,并将其初始化为 undefined」。

    这样还是解释不了我的疑惑,为什么明明是函数a先提升的,变量a后提升的,为什么打印出来的不是undefined?找了很多资料似乎也就那句话函数优先级要高于变量优先级,相同命名,变量会被覆盖理解不了啊!而且我觉得这句话有误导的嫌疑!
    上文提到过var 声明会在代码执行之前就将「创建变量,并将其初始化为 undefined」。但是,如果发现已经这个变量在前面赋值了,就不会有初始化这一步为undefined这一步,看代码:

       function a(){console.log('hello')};
       var a;
       console.log(a)//ƒ a(){console.log('hello')}
    
       function a(){console.log('hello')};
       var a = undefined;
       console.log(a)//undefined
    

    所以上述过程应该这样子的:

    进行变量提升,function a()被提升到作用域的最前面,变量a也进行提升了,但是它发现已经有a了,所以没有了初始化a为undefined这一步。所以在alert(a)的时候,依旧是 ƒ a(){alert('10')}

  2. 10 //执行了a(){alert(10)}
    不知道你看懂了吗????

条件判断下的函数提升

上面既然我们已经说到,对于函数声明(不是函数表达式哦)来讲,在代码执行之前,JS引擎就会对其进行创建,初始化,赋值。但是,遇到条件判断语句,像下面的:

(function () { 
    if (false) {
        function test() {
            console.log(2);
        }
    }
    test();
})();

在早期的浏览器中,会打印出2,因为对于函数声明来讲,在代码执行之前,JS引擎就会对其进行创建,初始化,赋值。所以就算进入条件判断语句,test()函数已经在当前作用域创建并赋值了。
但很明显,这跟我们的出发点是背离的,我们写的条件判断语句完全没有意义了!
后来浏览器为了修正这个错误,像这种情况下面的函数提升就只是 var test提升到函数作用域的顶端,本题中false所以没进入函数体,所以test()就会报错test is not a function。而不是像第一题那样,整个函数体都提到前面。
除了if,好像while,switch,for也一样。

ecma文档:http://www.ecma-international.org/ecma-262/6.0/index.html

再来做做题吧!

解释一下下面执行的结果

    (function(){
    console.log(a);
    function a(){
        console.log(a);
    }
    var a = 1;
    })() //ƒ a(){console.log(a);}

你看懂了吗??

相关文章

  • JS中的提升

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

  • 变量提升和函数提升的问题

    关于变量提升,大家可能都听的耳朵都起茧了。但是最近的一次讨论 ,又让我有了新的认识。 先看看题:请说出1,2对应序...

  • 2021-04-02

    变量提升和函数提升以及他们的优先级 1,变量提升:变量提升是指将变量声明提升到它所在的作用域的最开始部分 2,函数...

  • 变量提升问题

    这是 变量提升问题 1. 变量提升, 很简单,就是把变量提升提到函数的top的地方。我们需要说明的是,变量提升 只...

  • 引用类型

    变量提升和函数声明提升   函数声明提升就是把函数声明提升到函数声明所在作用域中(或者说一个函数体内)的顶端,变量...

  • 关于变量提升

    函数提升的优先级要高于变量提升,变量提升到函数声明的后面; 变量提升,但是赋值不提升,函数表达式不提升;

  • 变量提升

    变量提升 Hoisting 变量和函数的声明会在编译时被提升到作用域的顶部 提升的是声明,函数会连带函数体提升 i...

  • let const var

    变量提升 函数会优先于变量提升; 函数提升会把整个函数移到作用域顶部 变量提升智慧把变量的定义移到作用域顶部 wi...

  • js3-函数变量提升 顶层函数 数据类型转换 Math对象 js

    函数变量的提升 函数内部变量提升: 函数内部使用变量, 默认情况下, 函数会把下面变量它的声明提升到函数的最上边 ...

  • js的提升( hoisting ),以及var,let,cons

    1、 变量提升( var 和 function提升 ) 所谓“变量提升”,意味着变量和函数的声明会在物理层面移动到...

网友评论

      本文标题:变量提升和函数提升的问题

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