var、let和const之间的区别

简介: var、let和const之间的区别

作用域不同

var是函数作用域, letconst是块级作用域

  • 函数作用域就是在函数中声明了 var变量,那么这个变量在整个函数里都是有效的。
  • 块级作用域则是在块里是有效的。块级作用域就是用{}包住的区域,常用的有forwhileif等,只是有 {}包住也是块级作用域
{
  var a = 111
  let b = 222
  const c = 333

  console.log(a)  // 111
  console.log(b)// 222
  console.log(c)// 333
}

console.log(a)  // 111
// console.log(b) // b is not defined
console.log(c)  // c is not defined

有无变量提升

var有变量提升, let const没有变量提升

let const不需要先声明,再使用,否则会报错,而 var不需要先声明再使用,可以先使用后声明,不会报错,不过赋值的时候,值一直是 undefined

console.log(a)    // undefined
console.log(b)    // Cannot access 'b' before initialization
// console.log(c)  // Cannot access 'c' before initialization

var a = 111
let b = 222
// const c = 333

注意,这里报的错不是 b is not defined而是 Cannot access 'b' before initialization。也就是说:

  • 从广义上来说, let const没有变量提升,因为在声明前使用会报错
  • 从狭义上来说, let const是有变量提升的,因为实际上用它们定义的变量已经被执行上下文记住了,否则应该会报错 is not defined才对。

let const究竟有没有变量提升取决于怎么定义变量提升

  • 如果变量提升指的是变量可以在声明前使用,则没有变量提升
  • 如果变量提升指的是变量在声明前有没有被执行上下文记住的话,则是有变量提升的。

能否被重新定义

let const不能被重新声明,但是var可以被重新声明

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

// let b = 1
// let b = 2    // Identifier 'b' has already been declared

// const c = 1
// const c = 2     // Identifier 'b' has already been declared

在这个例子中,把下面的注释去掉后,就能够再次证明上面说的 let const 有没有变量提升是取决于怎么定义的。

image-20220411210448047

因为JavaScript是单线程的,如果预编译执行上下文没有记住用 let const定义的变量的话,按理来说会先输出a,才报错,但是不是,也就是说,实际上执行上下文在预编译的时候,也已经记住用 let const声明的变量了。

不能在函数内部重新声明参数

function mytest(args) {
    let args    // 报错
}

但是,如果是在函数里的另一个块级作用域里则可以重新声明参数

function mytest(args) {
  {
    let args = 111;
    console.log(args)    // 111

    {
      let args = 222;
      console.log(args)    // 222
    }

  }
}

mytest(123)

暂时性死区

块中用 let const声明的变量,在使用 let const声明变量之前,该变量都是不可用的。这在语法上称为暂时性死区(temporal dead zone)

var a = 111

{
  console.log(a)    // Cannot access 'a' before initialization
  let a = 222
}

上面的例子,就是典型的暂时性死区的例子。在上面有全局变量,在下面有块级变量,但是在中间则是谁都没法访问到。

全局作用域下是否会挂载到window对象

全局作用域下,使用 var声明的变量会被挂载到 window对象上,而使用 let const 则不会

var a = 111
console.log(window.a)   // 111

let b = 222
console.log(window.b)   // undefined

const c = 333
console.log(window.c)   // undefined

constlet的区别

const var的区别如上。 const let的区别就是const声明的是常量,声明后不能够修改

常见面试题

for (var i = 0; i < 5; i++) {
  setTimeout(function () {
    console.log(i)    // 输出5,5次
  }, 0)
}

因为setTimeout是宏任务(JS执行机制可参考之前的文章),所以执行输出语句时,for循环已经执行完成了,然后用 var声明的变量是函数作用域,所以会打印出5个5。可以尝试在for循环外打印i,能得到 5的结果

把上面的 var换成 let后,会依次输出0、1、2、3、4

for (var i = 0; i < 5; i++) {
  setTimeout(function () {
    console.log(i)    // 输出0、1、2、3、4
  }, 0)
}

因为 let是块级作用域,用 let声明的变量会被绑定到 setTimeout上,所以并不会受到外界的影响,输出的结果就会是正常的0、1、2、3、4。

另外,for循环还有一个特别的地方:设置循环变量的部分是一个父作用域,而循环体内是一个单独的子作用域

for (let i = 0; i < 1; i++) {
  let i = 222       // 这里不会报错,因为这里实际上是另一个子作用域
  console.log(i)    // 222
}

这里顺便再来回顾一下,暂时性死区。

for (let i = 0; i < 1; i++) {
  console.log(i)        // Cannot access 'i' before initialization
    
  let i = 222      
  console.log(i)   
}

参考链接:

目录
相关文章
|
文字识别 算法 API
视觉智能开放平台产品使用合集之uniapp框架如何使用阿里云金融级人脸识别
视觉智能开放平台是指提供一系列基于视觉识别技术的API和服务的平台,这些服务通常包括图像识别、人脸识别、物体检测、文字识别、场景理解等。企业或开发者可以通过调用这些API,快速将视觉智能功能集成到自己的应用或服务中,而无需从零开始研发相关算法和技术。以下是一些常见的视觉智能开放平台产品及其应用场景的概览。
394 0
|
Java
byte加byte居然是int了
为什么在Java中,byte加byte的结果不是byte,而是int?本文将带你从Java语言规范和JVM规范层面研究该问题。
229 3
byte加byte居然是int了
|
12月前
|
监控 测试技术 开发工具
移动端性能测试工具
移动端性能测试工具
862 2
|
分布式计算 安全 NoSQL
【专栏】大规模数据处理在数字化时代的重要性及挑战,包括洞察驱动决策、效率提升和创新机遇
【4月更文挑战第27天】本文探讨了大规模数据处理在数字化时代的重要性及挑战,包括洞察驱动决策、效率提升和创新机遇,同时面临数据量大、多样性、实时性和安全问题。文章介绍了Hadoop、Spark、NoSQL数据库及流处理技术(Kafka、Flink)等主流处理工具。在技术选型与实施中,强调明确业务需求、成本效益分析、兼容性及安全合规,并建议分阶段实施和持续优化。大规模数据处理技术是企业数据化转型的基础,影响决策效率和创新能力,企业应灵活选用合适的技术,构建高效安全的数据处理体系,以实现数据价值最大化。
462 3
|
JavaScript 前端开发
5款MicrosoftEdge扩展推荐(亲测有用 极力推荐
5款MicrosoftEdge扩展推荐(亲测有用 极力推荐
479 0
5款MicrosoftEdge扩展推荐(亲测有用 极力推荐
|
监控 架构师 API
深入浅出微服务架构设计
在软件开发领域,微服务架构因其灵活性和可扩展性而受到青睐。本文将通过浅显易懂的方式解析微服务的核心概念、设计原则及其在实际开发中的应用,帮助读者构建起对微服务架构的全面认识,并启发性地探讨如何在这一架构下寻找到技术与业务的平衡点。
183 3
|
设计模式 算法 调度
【C++】开始使用优先队列
优先队列的使用非常灵活,它适合于任何需要动态调整元素优先级和快速访问最高(或最低)优先级元素的场景。在使用时,需要注意其插入和删除操作的时间复杂度,以及如何根据实际需求选择合适的仿函数。
166 4
|
存储 NoSQL 中间件
GitHub数据库榜单第一:Redis核心原理实践PDF,点赞已过百万+
Redis是互联网技术领域使用最为广泛的存储中间件,它是「Remote DictionaryService」的首字母缩写,也就是「远程字典服务」。Redis 以其超高的性能、完美的文档、简洁易懂的源码和丰富的客户端库支持在开源中间件领域广受好评。国内外很多大型互联网公司都在使用Redis, 比如Twitter、YouPom、暴雪娱乐、Github、StackOverflow、 腾讯、阿里、京东、华为、新浪微博等等,很多中小型公司也都有应用。也可以说,对Redis的了解和应用实践已成为当下中高级后端开发者绕不开的必备技能。
|
存储 算法
算法思想总结:模拟算法
算法思想总结:模拟算法
179 1
|
SQL 安全 关系型数据库
【超详细】安全测试===sqlmap使用心得(零)
零、前言 这篇文章是学习Sqlmap的用法时做的笔记,记录了Sqlmap的常见、基础用法。
690 0