JavaScript拾遗 · 变量提升(声)

ES6之前,JavaScript的变量提升机制会将var与function定义的变量提升到它所在的作用域最开始的部分,且var所定义的变量只会被提前声明,而function所定义的方法既会被提前声明,也会被提前定义。下面依次举例说明。

0x01.var 关键字

使用var所定义的变量会被提前到当前作用域的开始部分进行声明,而赋值操作则留在原来的位置。下面例子说明了使用var定义的变量在不同的作用域下变量提升的结果。

console.log(a); // undefined
var a = 1;
console.log(a); // 1

function test(){
    console.log(b); // undefined
    var b = 2;
    console.log(b); // 2
}

test();

上面这一段代码中,第一个console.log(a)会输出undefined而不会直接抛出Error:a is not defined,第二个console.log(a)则正确输出a的值。执行test方法时,局部变量b的输出结果也与全局变量a相似,其原因就是JavaScript的解释器在执行代码时会将变量a与局部变量b的声明语句自动提升到该变量所处的作用域的最前端,按照下面的逻辑顺序执行:

var a;
console.log(a); // 此时a仅仅被定义而未被赋值,所以输出undefined
a = 1;
console.log(a); // 此时的a已经有值

function test(){
    var b;
    console.log(b); // 此时b仅仅被定义而未被赋值,所以输出undefined
    b = 2;
    console.log(b); // 此时的b已经有值
}

test();

0x02.function 关键字

使用function所定义的方法会被整个提升到当前作用域的最前端,下面举例说明:

console.log(test); // function test(){ console.log('test'); }

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

上面这一段代码控制台打印出了整个方法体的字符串,其原因就是使用function所定义的方法在JavaScript的解释器执行过程中将会提升到当前作用域的最前端来声明和定义,这里和使用var关键字有不同,var关键字仅仅只会提升变量的声明,下面的代码是真实的执行顺序:

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

console.log(test); // function关键字将整个函数体的声明与定义提升到了当前作用域的最前端,故控制台输出了整个函数体字符串。

0x03.使用function与var定义函数的对比

在之前使用JavaScript中我们通常有两种定义函数的方式,如下:

function test1(){}

var test2 = function(){}

按照之前所说var与function在对所定义的变量和方法进行提升时有些许不一样的地方,所以根据变量提升原则,两种定义方式的方式所带来的结果也是不一样的,使用function定义的方法在定义之前调用不会受到影响,而使用var定义的方法在定义之前进行调用则解释器会报错,如下:

test1(); // 可以顺利执行并打印出use keyword function

test2(); // 控制台抛出错误信息test2 is not a function

function test1(){
    console.log('use keyword function');
}

var test2 = function(){
    console.log('use keyword var');
}

 然后我们将代码还原为变量提升后的逻辑顺序再看,就能发现会造成结果不同的原因,正是因为使用function关键字定义的方法会被解释器整个提升到作用域最前端,而使用var定义的变量test2仅仅将定义部分提升到最前端,由于JavaScript弱类型的语言特性,此时的test2不是函数类型,故调用test2解释器会给出test2 is not a function的错误信息,所以在我们的日常开发中,为了减少BUG出现的几率,应当少使用或不使用var关键字来定义方法。

function test1(){
    console.log('use keyword function');
}

var test2;

test1(); // 此时test1方法已经被声明且定义,可以顺利执行

test2(); // 此时test2被声明,但未被定义,test2此时为undefined,还不能够被执行

test2 = function(){
    console.log('use keyword var');
}

 0x04.来自MDN的解释

变量提升(Hoisting)被认为是, Javascript中执行上下文 (特别是创建和执行阶段)工作方式的一种认识。您在 ECMAScript® 2015 Language Specification 之前的JavaScript文档中找不到变量提升(Hoisting)这个词。

不过,需要注意的是,这个概念可能产生一点点误解 。

例如,从概念的字面意义上说,“变量提升”意味着变量和函数的声明会在物理层面移动到代码的最前面,但这么说并不准确。实际上变量和函数声明在代码里的位置是不会动的,而是在编译阶段被放入内存中。

从上面MDN对变量提升的官方解释来看,变量提升并不会改变代码的物理位置,而将这个过程放在编译阶段放入内存,所以我们在开发者工具中看JavaScript代码是无法看到变量提升后的代码的。

0x05.参考资料

MDN web docs——变量提升

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 JavaScript 中,变量提升指的是在代码执行前,变量明就已经被提升(即复制)到了当前作用域的最顶端。这意味着,无论变量在代码中定义在哪里,它都会在代码执行前被提升到所在作用域的最顶端。 例如: ``` console.log(a); // undefined var a = 1; ``` 在这个例子中,虽然变量 `a` 实际上定义在了第二行,但是在代码执行前,它已经被提升到了最顶端。所以在第一行中,当我们尝试访问 `a` 时,它已经被明过了,所以输出 `undefined`。 注意,变量提升仅仅是明(复制)被提升,而不是赋值。所以在上面的例子中,变量 `a` 的值在第二行被赋值为 1,而不是在第一行被赋值为 `undefined`。 ### 回答2: JavaScript变量提升JavaScript中的一种特性。在JavaScript中,变量明会在代码执行之前进行处理,也就是说它们会被“提升”到其作用域的顶部。这意味着你可以在明之前使用变量,而不会引发错误。 例如,假设我们有以下的JavaScript代码: ``` console.log(x); // undefined var x = 5; console.log(x); // 5 ``` 在这个例子中,变量x被提升到了代码的顶部,所以在第一个console.log语句中,x被明了但尚未被赋值,所以它的值是undefined。然后,变量x被赋值为5,并在第二个console.log语句中打印出来。 需要注意的是,只有变量明会被提升,而不是它们的赋值。例如: ``` console.log(y); // ReferenceError: y is not defined y = 10; console.log(y); // 10 var y; ``` 在这个例子中,变量y在它的明之前被使用会引发错误。只有在变量y的明后,它的赋值才会生效。 变量提升在理解JavaScript的作用域和执行顺序时非常重要。它使得我们可以在变量明之前使用变量,但也可能导致一些意外的结果,所以在编写代码时要小心使用。 ### 回答3: JavaScript变量提升是指在代码执行之前,所有变量明都会被提升到代码的顶部。这意味着我们可以在变量明之前使用这些变量。但是要记住的是,只有变量明会被提升,而不是初始化。 例如,我们可以在变量明之前输出变量的值: console.log(x); // undefined var x = 5; 这段代码中,变量x在明之前被赋值为undefined,因此在打印x时,它的值是undefined。 变量提升还可以应用于函数明。我们可以在函数明之前调用函数: myFunction(); // "Hello, World!" function myFunction() { console.log("Hello, World!"); } 在这个例子中,函数myFunction在明之前被调用,所以我们可以在调用函数之前定义它。 需要注意的是,变量提升仅适用于使用var关键字明的变量,而不适用于使用let和const关键字明的变量。使用let和const明的变量是块级作用域的,不会被提升。 综上所述,JavaScript变量提升是一种将变量提升到代码顶部的机制,使我们可以在变量明之前使用这些变量。这在代码书写和阅读上提供了一定的便利,但也需要注意一些细节,以避免出现意料之外的问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值