ES6,ES2105核心功能一览,js新特性详解
过去几年 JavaScript 发生了很大的变化。ES6(ECMAScript 6、ES2105)是 JavaScript 语言的新标准,2015 年 6 月正式发布后,得到了迅速推广,使得JavaScript语言可以用来编写复杂的大型应用程序,成为企业级开发语言。
ES6中包含了许多新的语言特性,它们将使JS变得更加强大,更富表现力。ECMAScript涵盖了各种环境中JS的使用场景,无论是浏览器环境还是类似node.js的非浏览器环境。最常用的ES6特性:let, const, class, extends, super, arrow functions, template string, destructuring, default, rest argumentsBabel是一个广泛使用的ES6转码器,可以将ES6代码转为ES5代码,从而在现有环境执行。let, const 这两个的用途与var类似,都是用来声明变量的,但在实际运用中他俩都有各自的特殊用途。
ES5只有全局作用域和函数作用域,没有块级作用域,这带来很多不合理的场景。第一种场景就是你现在看到的内层变量覆盖外层变量。而let则实际上为JavaScript新增了块级作用域。用它所声明的变量,只在let命令所在的代码块内有效。另外一个var带来的不合理场景就是用来计数的循环变量泄露为全局变量。而使用let则不会出现这个问题。const也用来声明变量,但是声明的是常量。一旦声明,常量的值就不能改变。ES6提供了更接近传统语言的写法,引入了Class(类)这个概念。新的class写法让对象原型的写法更加清晰、更像面向对象编程的语法,也更加通俗易懂。
class, extends, super 这三个特性涉及了ES5中最令人头疼的的几个部分:原型、构造函数,继承。不用考虑那么多指针的问题了。箭头函数arrow function是ES6最最常用的一个新特性,用它来写function比原来的写法要简洁清晰很多:
function(i){ return i + 1; } //ES5(i) => i + 1 //ES6(x, y) => {x++; y--; return x+y}//方程比较复杂,则需要用{}把代码包起来模板字符串 template string,当我们要插入大段的html内容到文档中时,传统的写法非常麻烦,所以之前我们通常会引用一些模板工具库,比如mustache等等。
用反引号(\)来标识起始,用${}`来引用变量,而且所有的空格和缩进都会被保留在输出之中:<Link to={`/list/${zdz.name}`}>{zdz.name}</Link>解构 destructuring,ES6允许按照一定模式,从数组和对象中提取值,对变量进行赋值。
let dog = {type: 'animal', many: 2}let { type, many} = dogconsole.log(type, many) //animal 2默认值default意思就是默认值。传统的做法调用animal()方法时忘了传参数,就是加上这一句type = type || 'cat' 来指定默认值。
如果用ES6我们而已直接这么写:function animal(type = 'cat'){ console.log(type)}animal()展开操作符,之前使用arguments实现多参数传值,ES6用展开操作符 ... 达到相同的目的,例子:
function animals(...types){ console.log(types)}animals('cat', 'dog', 'fish') //["cat", "dog", "fish"]之前用 apply(),现在可以用展开操作符 ... :ES5:Math.max.apply(Math, [33,100,5,6,43]) // 100ES6:Math.max(...[33,100,5,6,43]) // 100=================================
JavaScript 自 1995 年面世以来,一直在缓慢地改进着。每隔几年都会有新的补充。1997 年成立的 ECMAScript 引领着 JavaScript 的发展。已发布的版本有 ES3、 ES5、 ES6 等。ES3 与 ES5 之间隔了 10 年,而ES5 与 ES6 之间隔了 6 年。改进的新模式是每年都渐进式地做一些小改动,而不是像 ES6 一样一次性地进行大量的更改。浏览器支持所有的现代浏览器和环境都已经支持 ES6 了。Chrome、MS Edge、Firefox、Safari、Node和其它很多环境都已经嵌入程序以支持 JavaScript ES6 的大部分功能。所以功能都可以直接使用。=================================
ES6 核心功能PS:可以写个测试页面或者在浏览器控制台上尝试下面的代码。
ES6:class Animal{ constructor(name){ this.name = name; } speak(){ console.log(this.name + ' makes a noise.'}; }}var animal = new Animal('animal');animal.speak();//animal makes a noise.
undefined-------------------------
ES5:var x = 'outer';function test(inner) { if (inner) { var x = 'inner'; // scope whole function return x; } return x; // gets redefined because line 4 declaration is hoisted}test(false); // undefinedtest(true); // inner块级作用域变量
在 ES6 中,我们使用 let/const,而非 var 来声明变量。var 有什么缺点呢?var 的问题是变量会泄露到其它代码块,比如 for 循环或是 if 块。test(false) 应该返回 outer,但并不是,得到的值是 undefined。为什么?
因为即使 if 块没有执行,第 4 行的表达式 var x 也被提升了。var 提升:
var 是在函数作用域中的,即使在声明前,它在整个函数内都是可用的。声明被提升了。所以变量在声明前就可以使用。初始化 没有 被提升。如果使用 var 的话,一定 要在顶部声明变量。应用提升规则后,就好理解代码是如何执行的了:ES5var x = 'outer';function test(inner) { var x; // HOISTED DECLARATION if (inner) { x = 'inner'; // INITIALIZATION NOT HOISTED return x; } return x;}ES6解决了这个问题:
let x = 'outer';function test(inner) { if (inner) { let x = 'inner'; return x; } return x; // gets result from line 1 as expected}test(false); // outertest(true); // inner用 let 替代 var 以便按预期的那样去执行代码。如果没有调用 if 块,变量 x 就不会提升到块外。
Let 提升 和“temporal dead zone”在 ES6 中,let 会将变量提升到块顶部(而 非 像 ES5 是函数顶部)。但是在变量声明前引用变量会造成 ReferenceError。let 是块作用域的,不可以在声明前使用。“Temporal dead zone”是块开始直到变量被声明的这段区域。IIFE在介绍 IIFE 之前,我们来看个例子:ES5{ var private = 1;}console.log(private); // 1如你所见,private 会发生泄漏。需要使用 IIFE(立即执行函数表达式)将其包起来:ES5(function(){ var private2 = 1;})();console.log(private2); // Uncaught ReferenceError如果看过 jQuery/lodash 或是其它开源项目的代码,你会注意到它们都利用了 IIFE,以避免污染全局环境,而只在全局下定义 _、$或是 jQuery。
ES6 更工整,不再需要使用 IIFE,只要用块和 let 就可以了:ES6{ let private3 = 1;}console.log(private3); // Uncaught ReferenceErrorConst 如果不希望变量的值再改变,可以使用 const。总之:用 let 和 const 代替 var。使用 const 进行引用;避免使用 var。
如果必须重新指定引用,可以用 let 代替 const。文本模板
遇到文本模板时,不必再用嵌套连接了。比如:ES5var first = 'Adrian';var last = 'Mejia';console.log('Your name is ' + first + ' ' + last + '.');现在可以用 反引号(`) 和字符串插值 ${}:ES6
const first = 'Adrian';const last = 'Mejia';console.log(`Your name is ${first} ${last}.`);多行字符串不必像这样再连接 + n 字符串了:ES5var template = '<li *ngFor="let todo of todos" [ngClass]="{completed: todo.isDone}" >n' +' <div class="view">n' +' <input class="toggle" type="checkbox" [checked]="todo.isDone">n' +' <label></label>n' +' <button class="destroy"></button>n' +' </div>n' +' <input class="edit" value="">n' +'</li>';console.log(template);ES6 中同样可以用反引号解决:
const template = `<li *ngFor="let todo of todos" [ngClass]="{completed: todo.isDone}" > <div class="view"> <input class="toggle" type="checkbox" [checked]="todo.isDone"> <label></label> <button class="destroy"></button> </div> <input class="edit" value=""></li>`;console.log(template);两段代码会得到完全相同的结果。解构赋值
ES6 解构非常简明并且好用。看看下面的例子:获取数组元素ES5var array = [1, 2, 3, 4];var first = array[0];var third = array[2];console.log(first, third); // 1 3等同于:
ES6const array = [1, 2, 3, 4];const [first, ,third] = array;console.log(first, third); // 1 3调换值
ES5var a = 1;var b = 2;var tmp = a;a = b;b = tmp;console.log(a, b); // 2 1等同于
ES6let a = 1;let b = 2;[a, b] = [b, a];console.log(a, b); // 2 1返回多个值的解构
ES5 function margin() { var left=1, right=2, top=3, bottom=4; return { left: left, right: right, top: top, bottom: bottom };}var data = margin();var left = data.left;var bottom = data.bottom;console.log(left, bottom); // 1 4在第3行,也可以像这样用数组返回(并保存序列):
return [left, right, top, bottom];但之后调用时需要考虑返回数据的顺序。var left = data[0];var bottom = data[3];ES6 中调用时只会选择需要的数据(第 6 行):
ES6function margin() { const left=1, right=2, top=3, bottom=4; return { left, right, top, bottom };}const { left, bottom } = margin();console.log(left, bottom); // 1 4注意:第3行用到了一些其它的 ES6 功能。可以将 { left: left } 简化为 { left }。看看和 ES5 的版本相比,现在多简洁啊~很酷不是吗?
参数匹配解构
ES5var user = {firstName: 'Adrian', lastName: 'Mejia'};function getFullName(user) { var firstName = user.firstName; var lastName = user.lastName; return firstName + ' ' + lastName;}console.log(getFullName(user)); // Adrian Mejia等同于(但更简洁):
ES6const user = {firstName: 'Adrian', lastName: 'Mejia'};function getFullName({ firstName, lastName }) { return `${firstName} ${lastName}`;}console.log(getFullName(user)); // Adrian Mejia深度匹配
ES5function settings() { return { display: { color: 'red' }, keyboard: { layout: 'querty'} };}var tmp = settings();var displayColor = tmp.display.color;var keyboardLayout = tmp.keyboard.layout;console.log(displayColor, keyboardLayout); // red querty等同于(但更简洁):
ES6function settings() { return { display: { color: 'red' }, keyboard: { layout: 'querty'} };}const { display: { color: displayColor }, keyboard: { layout: keyboardLayout }} = settings();console.log(displayColor, keyboardLayout); // red querty也叫对象解构。如你所见,解构很有用,并有助于形成好的编码风格。最佳实践:
使用数组解构获取元素或调换变量,这样就不用创建临时引用了。对于多返回值的情况,不要用数组解构,用对象解构。类和对象
ES6 用“类”替代“构造函数”。在 JavaScript 中,每个对象都有原型对象。所有 JavaScript 对象都从原型上继承方法和属性。ES5 以面向对象编程(OOP)的方式创建对象,是利用构造函数实现的:ES5var Animal = (function () { function MyConstructor(name) { this.name = name; } MyConstructor.prototype.speak = function speak() { console.log(this.name + ' makes a noise.'); }; return MyConstructor;})();var animal = new Animal('animal');animal.speak(); // animal makes a noise.ES6 提供了语法糖,可以用 class、constructor 等新的关键字、更少的样板代码实现相同的效果。同样可以看到相比于constructor.prototype.speak = function (),用 speak() 定义方法更加清晰:
ES6class Animal { constructor(name) { this.name = name; } speak() { console.log(this.name + ' makes a noise.'); }}const animal = new Animal('animal');animal.speak(); // animal makes a noise.可以看到两种方式(ES5/6)的结果和使用方式相同。最佳实践:
最好使用 class 语法,避免直接操作 prototype。原因是这样代码更加简明易懂。避免出现空的构造器。如果没有指明,类会有默认的构造器的。继承
基于前面的 Animal 类,现在想要拓展 Animal,定义一个 Lion 类。ES5 原型继承的方式有些复杂。ES5var Lion = (function () { function MyConstructor(name){ Animal.call(this, name); } // prototypal inheritance MyConstructor.prototype = Object.create(Animal.prototype); MyConstructor.prototype.constructor = Animal; MyConstructor.prototype.speak = function speak() { Animal.prototype.speak.call(this); console.log(this.name + ' roars '); }; return MyConstructor;})();var lion = new Lion('Simba');lion.speak(); // Simba makes a noise.// Simba roars.在此我没有详细解读所有的代码,但是需要注意:
第 3 行,明确地用参数调用 Animal 构造器。
7-8 行,将 Lion 原型赋值为 Animal的原型。11 行,从父类 Animal 调用了 speak 方法。ES6 提供了新的关键字 extends 和 super。ES6
class Lion extends Animal { speak() { super.speak(); console.log(this.name + ' roars '); }}const lion = new Lion('Simba');lion.speak(); // Simba makes a noise.// Simba roars.效果相同,但是相比于 ES5,ES6 代码更易读,完胜。
最佳实践:
使用内置的 extends 实现继承。原生 Promise用 promise 替代回调地狱ES5function printAfterTimeout(string, timeout, done){ setTimeout(function(){ done(string); }, timeout);}printAfterTimeout('Hello ', 2e3, function(result){ console.log(result); // nested callback printAfterTimeout(result + 'Reader', 2e3, function(result){ console.log(result); });});这个函数接收一个回调,在 done 后执行。我们想要先后执行两次,所以在回调中又一次调用了 printAfterTimeout。
如果需要 3 或 4 次回调,代码很快就一团糟了。那么用 promise 如何实现呢:ES6function printAfterTimeout(string, timeout){ return new Promise((resolve, reject) => { setTimeout(function(){ resolve(string); }, timeout); });}printAfterTimeout('Hello ', 2e3).then((result) => { console.log(result); return printAfterTimeout(result + 'Reader', 2e3);}).then((result) => { console.log(result);});promise 中可以用 then 在某个函数完成后执行新的代码,而不必再嵌套函数。
箭头函数
ES6 没有移除函数表达式,但是新增了箭头函数。ES5 中,this 的指向有问题:ES5var _this = this; // need to hold a reference$('.btn').click(function(event){ _this.sendData(); // reference outer this});$('.input').on('change',function(event){ this.sendData(); // reference outer this}.bind(this)); // bind to outer this在函数内,需要用临时变量指向 this 或者使用 bind 绑定。ES6 中可以使用箭头函数。
For…of
最开始用 for ,然后使用 forEach,而现在可以用 for…of:ES6// this will reference the outer one$('.btn').click((event) => this.sendData());// implicit returnsconst ids = [291, 288, 984];const messages = ids.map(value => `ID is ${value}`);ES6 的 for…of 也可以用来迭代。// 向控制台输出对象的可枚举属性for (var key of Object.keys(someObject)) { console.log(key + ": " + someObject[key]);}默认参数
之前需要检测变量是否定义了,而现在可以指定 default parameters 的值。或许你之前像下面这样写过?ES5function point(x, y, isFlag){ x = x || 0; y = y || -1; isFlag = isFlag || true; console.log(x,y, isFlag);}point(0, 0) // 0 -1 true point(0, 0, false) // 0 -1 true point(1) // 1 -1 truepoint() // 0 -1 true这可能是检测变量有值或指定默认值的惯用模式,但也存在一些问题:
第 8 行,我们传的值是 0, 0 但是得到的是 0, -1第 9 行,传进去 false 但是得到的是 true。如果默认参数是布尔值或将值设为 0,是没有用的。想知道为什么?我会在下面的 ES6 示例后说明。有了 ES6,现在可以用更少的代码实现更好的效果了。ES6function point(x = 0, y = -1, isFlag = true){ console.log(x,y, isFlag);}point(0, 0) // 0 0 truepoint(0, 0, false) // 0 0 falsepoint(1) // 1 -1 truepoint() // 0 -1 true注意第 5 行和第 6 行我们拿到了想要的值。ES5 的示例不好用,是因为先要检测 undefined的值,而 false、 null、 undefined 和 0 都是假的值。我们可以加些代码:
ES5function point(x, y, isFlag){ x = x || 0; y = typeof(y) === 'undefined' ? -1 : y; isFlag = typeof(isFlag) === 'undefined' ? true : isFlag; console.log(x,y, isFlag);}point(0, 0) // 0 0 truepoint(0, 0, false) // 0 0 falsepoint(1) // 1 -1 truepoint() // 0 -1 true现在当检测 undefined 值时就符合我们的要求了。剩余参数
之前使用 arguments,而现在可以用展开操作符。ES5 中处理不定参数很麻烦:ES5function printf(format) { var params = [].slice.call(arguments, 1); console.log('params: ', params); console.log('format: ', format);}printf('%s %d %.2f', 'adrian', 321, Math.PI);现在可以用展开操作符 ... 达到相同的目的。
ES6
function printf(format, ...params) { console.log('params: ', params); console.log('format: ', format);}printf('%s %d %.2f', 'adrian', 321, Math.PI);展开操作符
之前用 apply(),现在可以方便地使用展开操作符 ... 了:提示:apply() 可以将数组转化为一系列参数。例如 Math.max() 接收一系列参数,但如果想应用于数组的话可以用 apply 帮助实现。如上所述,apply 可以将数组当作参数序列进行传递:ES5Math.max.apply(Math, [2,100,1,6,43]) // 100ES6 可以用展开操作符:ES6Math.max(...[2,100,1,6,43]) // 100之前用 concat 合并数组,现在也可以用展开操作符:
ES5var array1 = [2,100,1,6,43];var array2 = ['a', 'b', 'c', 'd'];var array3 = [false, true, null, undefined];console.log(array1.concat(array2, array3));ES6 可以用展开操作符展开嵌套的数组:
ES6const array1 = [2,100,1,6,43];const array2 = ['a', 'b', 'c', 'd'];const array3 = [false, true, null, undefined];console.log([...array1, ...array2, ...array3]);参考英文原文:
================================
ES6是一次重大的版本升级,与此同时,由于ES6秉承着最大化兼容已有代码的设计理念,你过去编写的JS代码将继续正常运行。事实上,许多浏览器已经支持部分ES6特性,并将继续努力实现其余特性。这意味着,在一些已经实现部分特性的浏览器中,你的JS代码已经可以正常运行。如果到目前为止你尚未遇到任何兼容性问题,那么你很有可能将不会遇到这些问题,浏览器正飞速实现各种新特性。ECMAScript是一门充满活力的语言,并在不断进化中。
未来版本的规范中将持续进行重要的技术改进。ES6将彻底改变你编写JS代码的方式!