30 分钟掌握 ES6/ES2015 核心内容(上)

ECMAScript 6(以下简称 ES6)是 JavaScript 语言的下一代标准。因为当前版本的 ES6 是在 2015 年发布的,所以又称 ECMAScript 2015。

也就是说,ES6 就是 ES2015。

虽然目前并不是所有浏览器都能兼容 ES6 全部特性,但越来越多的程序员在实际项目当中已经开始使用 ES6 了。所以就算你现在不打算使用 ES6,但为了看懂别人的你也该懂点 ES6 的语法了…

在我们正式讲解 ES6 语法之前,我们得先了解下 Babel。

Babel

Babel 是一个广泛使用的 ES6 转码器,可以将 ES6 代码转为 ES5 代码,从而在现有环境执行。大家可以选择自己习惯的工具来使用 Babel,具体过程可直接在 Babel 官网查看:

最常用的 ES6 特性

let, const, class, extends, super, arrow functions, template string, destructuring, default, rest arguments

这些是 ES6 最常用的几个语法,基本上学会它们,我们就可以走遍天下都不怕啦!我会用最通俗易懂的语言和例子来讲解它们,保证一看就懂,一学就会。

let, const

这两个的用途与 var 类似,都是用来声明变量的,但在实际运用中他俩都有各自的特殊用途。

首先来看下面这个例子:

1
2
3
4
5
6
7
var name = "zach";
while (true) {
var name = "obama";
console.log(name); //obama
break;
}
console.log(name); //obama

使用 var 两次输出都是 obama,这是因为 ES5 只有全局作用域和函数作用域,没有块级作用域,这带来很多不合理的场景。第一种场景就是你现在看到的内层变量覆盖外层变量。而 let 则实际上为 JavaScript 新增了块级作用域。用它所声明的变量,只在 let 命令所在的代码块内有效。

1
2
3
4
5
6
7
let name = "zach";
while (true) {
let name = "obama";
console.log(name); //obama
break;
}
console.log(name); //zach

另外一个 var 带来的不合理场景就是用来计数的循环变量泄露为全局变量,看下面的例子:

1
2
3
4
5
6
7
var a = [];
for (var i = 0; i < 10; i++) {
a[i] = function() {
console.log(i);
};
}
a[6](); // 10

上面代码中,变量 i 是 var 声明的,在全局范围内都有效。所以每一次循环,新的 i 值都会覆盖旧值,导致最后输出的是最后一轮的 i 的值。而使用 let 则不会出现这个问题。

1
2
3
4
5
6
7
var a = [];
for (let i = 0; i < 10; i++) {
a[i] = function() {
console.log(i);
};
}
a[6](); // 6

再来看一个更常见的例子,了解下如果不用 ES6,而用闭包如何解决这个问题。

1
2
3
4
5
6
var clickBoxs = document.querySelectorAll(".clickBox");
for (var i = 0; i < clickBoxs.length; i++) {
clickBoxs[i].onclick = function() {
console.log(i);
};
}

我们本来希望的是点击不同的 clickBox,显示不同的 i,但事实是无论我们点击哪个 clickBox,输出的都是 5。下面我们来看下,如何用闭包搞定它。

1
2
3
4
5
6
7
8
9
10
function iteratorFactory(i) {
var onclick = function(e) {
console.log(i);
};
return onclick;
}
var clickBoxs = document.querySelectorAll(".clickBox");
for (var i = 0; i < clickBoxs.length; i++) {
clickBoxs[i].onclick = iteratorFactory(i);
}

const 也用来声明变量,但是声明的是常量。一旦声明,常量的值就不能改变。

1
2
const PI = Math.PI;
PI = 23; //Module build failed: SyntaxError: /es6/app.js: "PI" is read-only

当我们尝试去改变用 const 声明的常量时,浏览器就会报错。

const 有一个很好的应用场景,就是当我们引用第三方库的时声明的变量,用 const 来声明可以避免未来不小心重命名而导致出现 bug:

1
const monent = require("moment");

class, extends, super

这三个特性涉及了 ES5 中最令人头疼的的几个部分:原型、构造函数,继承…你还在为它们复杂难懂的语法而烦恼吗?你还在为指针到底指向哪里而纠结万分吗?

有了 ES6 我们不再烦恼!

ES6 提供了更接近传统语言的写法,引入了 Class(类)这个概念。新的 class 写法让对象原型的写法更加清晰、更像面向对象编程的语法,也更加通俗易懂。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
class Animal {
constructor() {
this.type = "animal";
}
says(say) {
console.log(this.type + " says " + say);
}
}
let animal = new Animal();
animal.says("hello"); //animal says hello
class Cat extends Animal {
constructor() {
super();
this.type = "cat";
}
}
let cat = new Cat();
cat.says("hello"); //cat says hello

上面代码首先用 class 定义了一个“类”,可以看到里面有一个 constructor 方法,这就是构造方法,而 this 关键字则代表实例对象。简单地说,constructor 内定义的方法和属性是实例对象自己的,而 constructor 外定义的方法和属性则是所有实例对象可以共享的。

Class 之间可以通过extends关键字实现继承,这比 ES5 的通过修改原型链实现继承,要清晰和方便很多。上面定义了一个 Cat 类,该类通过 extends 关键字,继承了 Animal 类的所有属性和方法。

super关键字,它指代父类的实例(即父类的 this 对象)。子类必须在 constructor 方法中调用 super 方法,否则新建实例时会报错。这是因为子类没有自己的 this 对象,而是继承父类的 this 对象,然后对其进行加工。如果不调用 super 方法,子类就得不到 this 对象。

ES6 的继承机制,实质是先创造父类的实例对象 this(所以必须先调用 super 方法),然后再用子类的构造函数修改 this。

P.S 如果你写 react 的话,就会发现以上三个东西在最新版 React 中出现得很多。创建的每个 component 都是一个继承 React.Component 的类。详见 react 文档

arrow function

这个恐怕是 ES6 最最常用的一个新特性了,用它来写 function 比原来的写法要简洁清晰很多:

1
2
3
4
function(i){
return i + 1;
} //ES5
(i) => i + 1//ES6

简直是简单的不像话对吧…

如果方程比较复杂,则需要用{}把代码包起来:

1
2
3
4
5
6
7
8
9
10
function(x, y) {
x++;
y--;
return x + y;
}
(x, y) => {
x++;
y--;
return x+y
}

除了看上去更简洁以外,arrow function 还有一项超级无敌的功能!

长期以来,JavaScript 语言的 this 对象一直是一个令人头痛的问题,在对象方法中使用 this,必须非常小心。例如:

1
2
3
4
5
6
7
8
9
10
11
12
class Animal {
constructor() {
this.type = "animal";
}
says(say) {
setTimeout(function() {
console.log(this.type + " says " + say);
}, 1000);
}
}
var animal = new Animal();
animal.says("hi"); //undefined says hi

运行上面的代码会报错,这是因为 setTimeout 中的 this 指向的是全局对象。所以为了让它能够正确的运行,传统的解决方法有两种:

  1. 第一种是将 this 传给 self,再用 self 来指代 this

    1
    2
    3
    4
    5
    6
    says(say){
    var self = this;
    setTimeout(function(){
    console.log(self.type + ' says ' + say)
    }, 1000)
    }
  2. 第二种方法是用 bind(this),即

    1
    2
    3
    4
    5
    says(say){
    setTimeout(function(){
    console.log(this.type + ' says ' + say)
    }.bind(this), 1000)
    }

    但现在我们有了箭头函数,就不需要这么麻烦了:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    class Animal {
    constructor() {
    this.type = "animal";
    }
    says(say) {
    setTimeout(() => {
    console.log(this.type + " says " + say);
    }, 1000);
    }
    }
    var animal = new Animal();
    animal.says("hi"); //animal says hi

    当我们使用箭头函数时,函数体内的 this 对象,就是定义时所在的对象,而不是使用时所在的对象。
    并不是因为箭头函数内部有绑定 this 的机制,实际原因是箭头函数根本没有自己的 this,它的 this 是继承外面的,因此内部的 this 就是外层代码块的 this。

template string

这个东西也是非常有用,当我们要插入大段的 html 内容到文档中时,传统的写法非常麻烦,所以之前我们通常会引用一些模板工具库,比如 mustache 等等。

大家可以先看下面一段代码:

1
2
3
4
5
6
7
8
9
$("#result").append(
"There are " +
basket.count +
" " +
"items in your basket, " +
"" +
basket.onSale +
" are on sale!"
);

我们要用一堆的’+’号来连接文本与变量,而使用 ES6 的新特性模板字符串``后,我们可以直接这么来写:

1
2
3
4
5
$("#result").append(
`There are ${basket.count} items in your basket, ${
basket.onSale
} are on sale!`
);

用反引号(\)来标识起始,用 ${} 来引用变量,而且所有的空格和缩进都会被保留在输出之中,是不是非常爽?!

React Router 从第 1.0.3 版开始也使用 ES6 语法了,比如这个例子:

1
2
3
<Link to={`/taco/${taco.name}`}>{taco.name}Link>

React Router

destructuring

ES6 允许按照一定模式,从数组和对象中提取值,对变量进行赋值,这被称为解构(Destructuring)。

看下面的例子:

1
2
3
4
let cat = "ken";
let dog = "lili";
let zoo = { cat: cat, dog: dog };
console.log(zoo); //Object {cat:"ken", dog:"lili"}

用 ES6 完全可以像下面这么写:

1
2
3
4
let cat = "ken";
let dog = "lili";
let zoo = { cat, dog };
console.log(zoo); //Object {cat:"ken", dog:"lili"}

反过来可以这么写:

1
2
3
let dog = { type: "animal", many: 2 };
let { type, many } = dog;
console.log(type, many); //animal 2

default, rest

default 很简单,意思就是默认值。大家可以看下面的例子,调用 animal()方法时忘了传参数,传统的做法就是加上这一句 type = type || ‘cat’ 来指定默认值。

1
2
3
4
5
function animal(type) {
type = type || "cat";
console.log(type);
}
animal();

如果用 ES6 我们而已直接这么写:

1
2
3
4
function animal(type = "cat") {
console.log(type);
}
animal();

最后一个 rest 语法也很简单,直接看例子:

1
2
3
4
function animals(...types) {
console.log(types);
}
animals("cat", "dog", "fish"); //["cat", "dog", "fish"]

而如果不用 ES6 的话,我们则得使用 ES5 的 arguments。

总结

以上就是 ES6 最常用的一些语法,可以说这 20%的语法,在 ES6 的日常使用中占了 80%…

30 分钟掌握 ES6/ES2015 核心内容(下)

import export

这两个家伙对应的就是 es6 自己的 module 功能。

我们之前写的 Javascript 一直都没有模块化的体系,无法将一个庞大的 js 工程拆分成一个个功能相对独立但相互依赖的小工程,再用一种简单的方法把这些小工程连接在一起。

这有可能导致两个问题:

  1. 一方面 js 代码变得很臃肿,难以维护
  2. 另一方面我们常常得很注意每个 script 标签在 html 中的位置,因为它们通常有依赖关系,顺序错了可能就会出 bug
    在 es6 之前为解决上面提到的问题,我们得利用第三方提供的一些方案,主要有两种 CommonJS(服务器端)和 AMD(浏览器端,如 require.js)。

如果想了解更多 AMD,尤其是 require.js,可以参看这个教程:why modules on the web are useful and the mechanisms that can be used on the web today to enable them

而现在我们有了 es6 的 module 功能,它实现非常简单,可以成为服务器和浏览器通用的模块解决方案。

ES6 模块的设计思想,是尽量的静态化,使得编译时就能确定模块的依赖关系,以及输入和输出的变量。CommonJS 和 AMD 模块,都只能在运行时确定这些东西。

上面的设计思想看不懂也没关系,咱先学会怎么用,等以后用多了、熟练了再去研究它背后的设计思想也不迟!好,那我们就上代码…

传统的写法

首先我们回顾下 require.js 的写法。假设我们有两个 js 文件: index.js 和 content.js,现在我们想要在 index.js 中使用 content.js 返回的结果,我们要怎么做呢?

首先定义:

1
2
3
4
//content.js
define("content.js", function() {
return "A cat";
});

然后 require:

1
2
3
4
//index.js
require(["./content.js"], function(animal) {
console.log(animal); //A cat
});

那 CommonJS 是怎么写的呢?

1
2
3
//index.js
var animal = require("./content.js"); //content.js
module.exports = "A cat";

ES6 的写法

1
2
3
//index.js
import animal from "./content"; //content.js
export default "A cat";

以上我把三者都列出来了,妈妈再也不用担心我写混淆了…

ES6 module 的其他高级用法

1
2
3
4
5
6
//content.js
export default "A cat";
export function say() {
return "Hello!";
}
export const type = "dog";

上面可以看出,export 命令除了输出变量,还可以输出函数,甚至是类(react 的模块基本都是输出类)

1
2
3
4
//index.js
import { say, type } from "./content";
let says = say();
console.log(`The ${type} says ${says}`); //The dog says Hello

这里输入的时候要注意:大括号里面的变量名,必须与被导入模块(content.js)对外接口的名称相同。

如果还希望输入 content.js 中输出的默认值(default), 可以写在大括号外面。

1
2
3
4
//index.js
import animal, { say, type } from "./content";
let says = say();
console.log(`The ${type} says ${says} to ${animal}`); //The dog says Hello to A cat

修改变量名

此时我们不喜欢 type 这个变量名,因为它有可能重名,所以我们需要修改一下它的变量名。在 es6 中可以用 as 实现一键换名。

1
2
3
4
//index.js
import animal, { say, type as animalType } from "./content";
let says = say();
console.log(`The ${animalType} says ${says} to ${animal}`); //The dog says Hello to A cat

模块的整体加载

除了指定加载某个输出值,还可以使用整体加载,即用星号(*)指定一个对象,所有输出值都加载在这个对象上面。

1
2
3
4
//index.js
import animal, * as content from "./content";
let says = content.say();
console.log(`The ${content.type} says ${says} to ${animal}`); //The dog says Hello to A cat

通常星号*结合 as 一起使用比较合适。

终极秘籍

考虑下面的场景:上面的 content.js 一共输出了三个变量(default, say, type),假如我们的实际项目当中只需要用到 type 这一个变量,其余两个我们暂时不需要。我们可以只输入一个变量:

1
import { type } from "./content";

由于其他两个变量没有被使用,我们希望代码打包的时候也忽略它们,抛弃它们,这样在大项目中可以显著减少文件的体积。

ES6 帮我们实现了!

不过,目前无论是 webpack 还是 browserify 都还不支持这一功能…

如果你现在就想实现这一功能的话,可以尝试使用 rollup.js

他们把这个功能叫做 Tree-shaking,哈哈哈,意思就是打包前让整个文档树抖一抖,把那些并未被依赖或使用的东西统统抖落下去。。。

看看他们官方的解释吧:

Normally if you require a module, you import the whole thing. ES2015 lets you just import the bits you need, without mucking around with custom builds. It’s a revolution in how we use libraries in JavaScript, and it’s happening right now.

注:转发自https://segmentfault.com/a/1190000004365693?share_user=1030000013416310