在介绍let和const之前我们先复习一下相关的知识点。

关于函数作用域

开发过程中,在ES6(ECMA2015)标准推出之前,声明变量的方式一直都是var,而变量的作用域一般也只在函数内部,即函数作用域。

function a () {
var test = 1;
}
console.log(test); //error, test is not defined

上面的实例反应了当我们定义在函数a中定义变量test的时候,其作用域只在a函数中,在函数a外面访问test,则会抛错。

关于变量提升

在javascript中,使用var声明的变量,不论是在什么地方声明的,都会提升到当前作用域的最顶部进行初始化,这种行为叫做变量提升

function test() {
console.log('1:', a); //undefined
if (false) {
var a = 1;
}
console.log('2:', a); //undefined
}
test();

关于上面的代码,两次访问a都是undefined,说明在作用域中实际上是能访问到a的,只不过未赋值,即使判断语句是false,第一个日志访问的a也不是is not defined,出现这种情况的原因实际上就是因为变量提升。

把上面的代码翻译成浏览器真正执行的代码实际上是:

function test() {
// 变量提升,对a进行初始化,但未赋值
var a;
console.log('1:', a); //undefined
if (false) {
a = 1;
}
console.log('2:', a); //undefined
}

由于变量提升的关系,因此在es6标准推出之前,javascript中实际上只有函数作用域,而没有块作用域({}),这里的块作用域可以简单的看做用{}包起来的代码。

let

let和const都能够声明块作用域,用法和var是类似的,特别都不会进行变量提升,而是被锁在当前的代码块中。

function test() {
if (true) {
console.log(a); // TDZ,俗称临时死区,用来描述变量不提升的现象
let a = 1;
}
}
test(); // a is not defined
function test() {
if (true) {
let a = 1;
}
console.log(a);
}
test(); // a is not defined

上面的两个例子都用let进行变量声明,第一个例子反映出来let声明的变量是不会出现变量提升现象的,而第二个例子说明let声明的变量只在块作用域中生效,那么正确的使用方法应该是:在块作用域中,先声明,在访问

function test() {
if (true) {
let a = 1;
console.log(a);
}
}
test() // a

const

const和let类似,用来声明常量,不过和let的区别则是,一旦声明则不可以更改,而且必须进行初始化赋值。下面的例子很清楚来说明这一点。

const type = 'aaa';
type = 'bbb';
console.log(type); // "type" is read-only const type;
type = 'aaa';
console.log(type); // Missing initializer in const declaration

我们再来看看下面这个例子

const type = 'aaa';
let type = 'bbb';
console.log(type); // Identifier 'type' has already been declared

同样出现的报错,实际上我们不论是使用const 还是 let都不能这样重复声明相同名称的变量,但是使用var是可以的,原因可以简单的理解为因为变量提升的存在,导致声明只有一次,赋值有两次。

const最重要的特点是一点声明,不可以修改,但是假如定义的是对象Object,那么可以修改对象内部的属性值,原因是因为当我们修改对象内部的属性值的时候对象的内存地址实际上是没有变化的,但是假如我们直接修改对象本身,则会出现报错,在这种情况下需要使用let。

const type = {
a: 1,
} type.a = 2 //没有直接修改type值,对象的指针不变,只是修改了type.a的属性值,这是允许的。
console.log(type) // {a: 2} type = { // 直接修改type本身,对象的内存地址会发生变化,不允许。
a: 2,
}
console.log(type) // Identifier 'type' has already been declared

const 和 let 的异同点。

相同点:

  • const和let都只在当前的块作用域内有效
  • 不存在变量提升
  • 不能重复声明

    不同点:
  • const不能再赋值,let可以再赋值

临时死区(TDZ)

上面我们已经提到了TDZ的场景,那么,有什么用呢?答案就是没什么用。可以理解为一个便于说明的定义。

临时死区的意思是在当前作用域的块内,在声明变量前的区域叫做临时死区。

if (true) {
// 这块区域是TDZ
let a = 1;
}

块级作用域的使用场景

块级作用域有哪些使用场景呢?

在for循环中使用var声明的循环变量,会跳出循环污染当前的函数作用域

for (var i = 0; i < 5; i++) {
setTimeout(() => {
console.log(i); // 5, 5, 5, 5, 5
}, 0);
}
console.log(i) // 5, i跳出了循环污染当前函数作用域 for (let i = 0; i < 5; i++) {
setTimeout(() => {
console.log(i); // 0, 1, 2, 3, 4
}, 0);
}
console.log(i) // i is not defined i无法污染外部函数

全局作用于声明

如果在全局作用域使用let或者const声明,那么声明的变量本身就是全局变量,并不会挂在window下

let a = 1;
var b = 2;
console.log(a); // 1
console.log(window.a) // undefined
console.log(b); // 2
console.log(window.b) // 2

实践

在实际开发中,我们是使用var, let, 还是const取决于变量是否需要更新,不更新的变量用const,更新的变量用let。var能用的场景都可以使用let来替代,但是需要注意作用域的区别。

ES6学习笔记(1,let和const)的更多相关文章

  1. ES6学习笔记(一)——let和const

    1.ES6学习之let.const (1).var.let.const 变(常)量声明 ES5 只有全局作用域和函数作用域,没有块级作用域,这带来很多不合理的场景. 在ES6中let就诞生了,实际上它 ...

  2. es6学习笔记1 --let以及const

    let语句的基本用法:  1.let声明的变量为块级作用域,只在最近的{}里面有效,如果在外部引用就会报错. { let a = 10; var b = "hello" } ale ...

  3. ES6学习笔记(1)----let和const命令

    参考书<ECMAScript 6入门>http://es6.ruanyifeng.com/ let和const命令 let 总结1.声明变量基本使用方法与var 相同  不同点  a.在代 ...

  4. ES6学习笔记之 let与const

    在js中,定义变量时要使用var操作符,但是var有许多的缺点,如:一个变量可以重复声明.没有块级作用域.不能限制修改等. //缺点1:变量可以重复声明 var a=1; var a=2; conso ...

  5. ES6学习笔记<一> let const class extends super

    学习参考地址1  学习参考地址2 ECMAScript 6(以下简称ES6)是JavaScript语言的下一代标准.因为当前版本的ES6是在2015年发布的,所以又称ECMAScript 2015:也 ...

  6. es6学习笔记-class之一概念

    前段时间复习了面向对象这一部分,其中提到在es6之前,Javasript是没有类的概念的,只从es6之后出现了类的概念和继承.于是乎,花时间学习一下class. 简介 JavaScript 语言中,生 ...

  7. ES6学习笔记<五> Module的操作——import、export、as

    import export 这两个家伙对应的就是es6自己的 module功能. 我们之前写的Javascript一直都没有模块化的体系,无法将一个庞大的js工程拆分成一个个功能相对独立但相互依赖的小 ...

  8. ES6学习笔记<四> default、rest、Multi-line Strings

    default 参数默认值 在实际开发 有时需要给一些参数默认值. 在ES6之前一般都这么处理参数默认值 function add(val_1,val_2){ val_1 = val_1 || 10; ...

  9. ES6学习笔记<三> 生成器函数与yield

    为什么要把这个内容拿出来单独做一篇学习笔记? 生成器函数比较重要,相对不是很容易理解,单独做一篇笔记详细聊一聊生成器函数. 标题为什么是生成器函数与yield? 生成器函数类似其他服务器端语音中的接口 ...

  10. ES6学习笔记<二>arrow functions 箭头函数、template string、destructuring

    接着上一篇的说. arrow functions 箭头函数 => 更便捷的函数声明 document.getElementById("click_1").onclick = ...

随机推荐

  1. SELECT语句逻辑运行顺序,你知道吗?

    引言 这不是一个什么多深的技术问题.多么牛叉的编程能力.这跟一个人的开发能力也没有很必定的直接关系,可是知道这些会对你的SQL编写,排忧及优化上会有很大的帮助.它不是一个复杂的知识点.可是一个很基础的 ...

  2. .NET反编译之manager,base.AutoScaleMode修复

    使用反编译软件导出项目时,出现警告:设计器无法处理第X 行的代码:this.AutoScaleMode = AutoScaleMode.Font;方法"InitializeComponent ...

  3. android 程序更新(没有sdcard)

    今天总结一下我们写的程序需要更新(默认用户已经安装上旧版本了),而用户更新时却没有sdcard的情况,如果有sdcard,那就不用废话了... 先说一下软件更新的基本原理,从服务器下载一个程序的apk ...

  4. Cocos2d-x 处理双击事件的两种方法

    在cocos2d-x的开发过程中有些时候也是需要用到双击的事件处理,那么由于在cocos2d-x中没有实现对双击的事件的处理,那么我们就需要自己用代码实现. 下面介绍两种方式实现双击事件的处理. (一 ...

  5. JAVA学习目录

    开发环境以及IDE准备相关: 1.JAVA环境搭建 2.初次使用IntelliJ IDEA 3.IntelliJ IDEA界面设置 4.IntelliJ IDEA快捷键介绍 SprintBoot系列: ...

  6. TP框架中关于if、else 分支结构逻辑错误

    TP框架中关于if.else 分支结构逻辑错误 代码中没有任何错误 将注释往下一行就可以解决 造成问题的原因: TP框架中 想分配变量可以使用assign方法 在[模块]中: $this->as ...

  7. linux 打包和压缩文件

    打包成tar文件 tar -cf mydir.tar mydir/ 打包tar压缩成gz tar -czf mydir.tar.gz mydir/ 解压mydirtar文件 tar -xvf mydi ...

  8. [Swift A] - Using Swift with Cocoa and Objective-C--Mix and Match

    Swift与Objective-C的兼容能力允许你在同一个工程中同时使用两种语言.你可以用这种叫做“mix and match”的特性来开发基于混合语言的应用.使用Swfit的最新特性--“mix a ...

  9. Nutch相关框架安装使用最佳指南(转帖)

    Nutch相关框架安装使用最佳指南 Chinese installing and using instruction  -  The best guidance in installing and u ...

  10. iOS-仿智联字符图片验证码

    概述 随机字符组成的图片验证码, 字符位数可改变, 字符可斜可正排列. 详细 代码下载:http://www.demodashi.com/demo/10850.html 项目中有时候会有这种需求: 获 ...