ES6新特性一览

ES全称ECMAScript,ECMAScript是ECMA制定的标准化脚本语言。目前JavaScript使用的ECMAScript版本为ECMA-417。关于ECMA的最新资讯可以浏览 ECMA news查看。

ECMA规范最终由TC39敲定。TC39由包括浏览器厂商在内的各方组成,他们开会推动JavaScript提案沿着一条严格的发展道路前进。 从提案到入选ECMA规范主要有以下几个阶段:

  1. Stage 0: strawman——最初想法的提交。
  2. Stage 1: proposal(提案)——由TC39至少一名成员倡导的正式提案文件,该文件包括API事例。
  3. Stage 2: draft(草案)——功能规范的初始版本,该版本包含功能规范的两个实验实现。
  4. Stage 3: candidate(候选)——提案规范通过审查并从厂商那里收集反馈
  5. Stage 4: finished(完成)——提案准备加入ECMAScript,但是到浏览器或者Nodejs中可能需要更长的时间。

ES6新特性(2015)

ES6的特性比较多,在 ES5 发布近 6 年(2009-11 至 2015-6)之后才将其标准化。两个发布版本之间时间跨度很大,所以ES6中的特性比较多。 在这里列举几个常用的:

  • 模块化
  • 箭头函数
  • 函数参数默认值
  • 模板字符串
  • 解构赋值
  • 延展操作符
  • 对象属性简写
  • Promise
  • Let与Const

1.类(class)

对熟悉Java,object-c,c#等纯面向对象语言的开发者来说,都会对class有一种特殊的情怀。ES6 引入了class(类),让JavaScript的面向对象编程变得更加简单和易于理解。

class Animal {
    constructor(name, color) {

        this.name = name;
        this.color = color;

    }

    toString() {
        console.log('name:'+ this.name + ',color:' + this.color);

    }
}

var animal = new Animal('dog', 'white');
animal.toString();
console.log(animal.hasOwnProperty('name'));
console.log(animal.hasOwnProperty('toString'));
console.log(animal.__proto__.hasOwnProperty('toString'));

class Cat extends Animal {
    constructor(action) {
        // 如果没有置顶consructor,默认带super函数的constructor将会被添加、
        super('cat', 'white');
        this.action = action;
    }

    toString() {
        console.log(super.toString());
    }

}

var cat = new Cat('catch');
cat.toString();
console.log(cat instanceof Cat);
console.log(cat instanceof Animal);

2.模块化(Module)

ES5不支持原生的模块化,在ES6中模块作为重要的组成部分被添加进来。模块的功能主要由 export 和 import 组成。每一个模块都有自己单独的作用域,模块之间的相互调用关系是通过 export 来规定模块对外暴露的接口,通过import来引用其它模块提供的接口。同时还为模块创造了命名空间,防止函数的命名冲突。

导出(export)

ES6允许在一个模块中使用export来导出多个变量或函数。

导出变量
export var name = 'Rainbow'

ES6不仅支持变量的导出,也支持常量的导出。
export const sqrt =Math.sqrt;//导出常量

ES6将一个文件视为一个模块,上面的模块通过 export 向外输出了一个变量。一个模块也可以同时往外面输出多个变量。

var name = 'Rainbow';
var age = '24';

export {
  name,
  age
};
导出函数
export function myModule (someArg) {
    return someArg;
}
导入(import)

定义好模块的输出以后就可以在另外一个模块通过import引用。

import myModule from 'myModule';
import { name ,age } from 'test';
  1. 一条import 语句可以同时导入默认函数和其它变量。
    如:import defaultMethod, { otherMethod } from 'xxx.js';
  2. 可以为变量起别名
    如:import { otherMethod as om } from 'xxx.js'

3.箭头(Arrow)函数

这是ES6中最令人激动的特性之一。 =>不只是关键字function的简写,它还带来了其它好处。箭头函数与包围它的代码共享同一个 this,能帮你很好的解决this的指向问题。有经验的JavaScript开发者都熟悉诸如 var self = this;或 var that =this这种引用外围this的模式。但借助 =>,就不需要这种模式了。

箭头函数的结构

箭头函数的箭头=>之前是一个空括号、单个的参数名、或用括号括起的多个参数名,而箭头之后可以是一个表达式(作为函数的返回值),或者是用花括号括起的函数体(需要自行通过return来返回值,否则返回的是undefined)。

() => 1;

(a, b) => a + b;

() => ([1, 2]);

() => ({
    a: 1,
    b: 2
});

() => {
    alert()
}

setTimeout(() => {
    // to do
}, 500)

不论是箭头函数还是bind,每次被执行都返回的是一个新的函数引用,因此如果你还需要函数的引用去做一些别的事情(譬如卸载监听器),那么你必须自己保存这个引用

卸载监听器时的陷阱
错误的做法
class PauseMenu extends React.Component {
    componentWillMount() {
        AppStateIOS.addEventListener('change', this.onAppPaused.bind(this));
    }
    componentWillUnmount(){
        AppStateIOS.removeEventListener('change', this.onAppPaused.bind(this));
    }
    onAppPaused(event){}
}
正确的做法
class PauseMenu extends React.Component {
    constructor(props) {
        super(props)
        this._onAppPaused = this.onAppPaused.bind(this)
    }

    componentWillMount() {
        AppStateIOS.addEventListener('change', this._onAppPaused);
    }
    componentWillUnmount(){
        AppStateIOS.removeEventListener('change', this._onAppPaused);
    }
    onAppPaused(event){}
}

除上述的做法外,babel最新版本允许我们还可以这样做:

class PauseMenu extends React.Component {
    constructor(props) {
        super(props)
    }

    componentWillMount() {
        AppStateIOS.addEventListener('change', this.onAppPaused);
    }
    componentWillUnmount(){
        AppStateIOS.removeEventListener('change', this.onAppPaused);
    }
    onAppPaused = (event) => {}
}

需要注意的是:不论是bind还是箭头函数,每次被执行都返回的是一个新的函数引用,因此如果你还需要函数的引用去做一些别的事情(譬如卸载监听器),那么你必须自己保存这个引用。

4.ES6支持在定义函数的时候为其设置默认值:

function foo(width= 20, height = 50) {
    console.log(width, height)
}

ES5设置默认值:

function foo(width, height) {
    var w = width || 20;
    var h = height || 50;
    console.log(w, h)
}

这样写一般没问题,但当 参数的布尔值为false时,就会有问题了。比如,我们这样调用foo函数:

foo(0,'');

因为 0的布尔值为false,这样width的取值将是20。
所以说, 函数参数默认值不仅能是代码变得更加简洁而且能规避一些问题。

5.模板字符串

ES6支持 模板字符串,使得字符串的拼接更加的简洁、直观。

  1. 不使用模板字符串:
var str = 'hello, ' + name + ', my name is ' + myName;
  1. 使用模板字符串:
let str = `hello, ${name}, my name is ${myName}`;

在ES6中通过 ${}就可以完成字符串的拼接,只需要将变量放在大括号之中。
解构赋值语法是JavaScript的一种表达式,可以方便的从数组或者对象中快速提取值赋给定义的变量。

6.解构赋值

获取数组中的值

从数组中获取值并赋值到变量中,变量的顺序与数组中对象顺序对应。

var foo = ["one", "two", "three", "four"];
var [one, two, three] = foo;
console.log(one);
console.log(two);
console.log(three);
var [first, , , last] = foo;
console.log(first);
console.log(last);
var a, b; [a, b] = [1, 2];
console.log(a);
console.log(b);

如果没有从数组中的获取到值,你可以为变量设置一个默认值。

var a, b; [a = 5, b = 7] = [1];
console.log(a);
console.log(b);

通过解构赋值可以方便的交换两个变量的值。

var a = 1;
var b = 3; [a, b] = [b, a];
console.log(a);
console.log(b);
获取对象中的值
const student = {
  name: 'Ming',
  age: '18',
  city: 'Shanghai'
};
const {
  name,
  age,
  city
} = student;
console.log(name);
console.log(age);
console.log(city);

7.延展操作符(Spread operator)

延展操作符...可以在函数调用/数组构造时, 将数组表达式或者string在语法层面展开;还可以在构造对象时, 将对象表达式按key-value的方式展开。

语法
  1. 函数调用:
myFunction(...iterableObj);
  1. 数组构造或字符串:
[...iterableObj, '4', ...'hello', 6];
  1. 构造对象时,进行克隆或者属性拷贝(ECMAScript 2018规范新增特性):
let objClone = { ...obj };
应用场景
  1. 在函数调用时使用延展操作符
function sum(x, y, z) {
  return x + y + z
}
const numbers = [1, 2, 3];
console.log(sum.apply(null, numbers));
console.log(sum(...numbers));
  1. 构造数组
    没有展开语法的时候,只能组合使用 push,splice,concat 等方法,来将已有数组元素变成新数组的一部分。有了展开语法, 构造新数组会变得更简单、更优雅:
const stuendts = ['Jine', 'Tom'];
const persons = ['Tony', ...stuendts, 'Aaron', 'Anna'];
conslog.log(persions)

和参数列表的展开类似, ... 在构造字数组时, 可以在任意位置多次使用。

  • 数组拷贝
var arr = [1, 2, 3];
var arr2 = [...arr];
arr2.push(4);
console.log(arr2)

展开语法和 Object.assign() 行为一致, 执行的都是浅拷贝(只遍历一层)。

  • 连接多个数组
var arr1 = [0, 1, 2];
var arr2 = [3, 4, 5];
var arr3 = [...arr1, ...arr2];
var arr4 = arr1.concat(arr2);

在ECMAScript 2018中延展操作符增加了对对象的支持

var obj1 = {
  foo: 'bar',
  x: 42
};
var obj2 = {
  foo: 'baz',
  y: 13
};
var clonedObj = { ...obj1 };
var mergedObj = { ...obj1, ...obj2 };
在React中的应用

通常我们在封装一个组件时,会对外公开一些 props 用于实现功能。大部分情况下在外部使用都应显示的传递 props 。但是当传递大量的props时,会非常繁琐,这时我们可以使用 ...(延展操作符,用于取出参数对象的所有可遍历属性) 来进行传递。

一般情况下我们应该这样写
<CustomComponent name='Jine' age={21}/>
  1. 使用 ... ,等同于上面的写法
const params = {
  name: 'Jine',
  age: 21
} 
< CustomComponent {...params} />
  1. 配合解构赋值避免传入一些不需要的参数
var params = {
  name: '123',
  title: '456',
  type: 'aaa'
}
var {
  type, ...other
} = params; 
<CustomComponent type = 'normal'number = {2} {...other} />
<CustomComponent type='normal'number={2}name='123'title='456' /> 

8.对象属性简写

在ES6中允许我们在设置一个对象的属性的时候不指定属性名。

  1. 不使用ES6
const name = 'Ming',
age = '18',
city = 'Shanghai';
const student = {
  name: name,
  age: age,
  city: city
};
console.log(student);

对象中必须包含属性和值,显得非常冗余。

  1. 使用ES6
const name = 'Ming',
age = '18',
city = 'Shanghai';
const student = {
  name,
  age,
  city
};
console.log(student);

对象中直接写变量,非常简洁。

9.Promise 是异步编程的一种解决方案,比传统的解决方案callback更加的优雅。它最早由社区提出和实现的,ES6 将其写进了语言标准,统一了用法,原生提供了Promise对象。

  1. 不使用ES6
    嵌套两个setTimeout回调函数:
setTimeout(function() {
  console.log('Hello');
  setTimeout(function() {
    console.log('Hi')
  },
  1000)
},
1000);
  1. 使用ES6
var waitSecond = new Promise(function(resolve, reject) {
  setTimeout(resolve, 1000)
});
waitSecond.then(function() {
  console.log("Hello");
  return waitSecond
}).then(function() {
  console.log("Hi")
});

10.支持let与const

在之前JS是没有块级作用域的,const与let填补了这方便的空白,const与let都是块级作用域。

  1. 使用var定义的变量为函数级作用域:
{
  var a = 10
}
console.log(a);
  1. 使用let与const定义的变量为块级作用域
{
  let a = 10
}
console.log(a); // 报错

下一篇文章会整理ES7、ES8、ES9、ES10新特性,敬请关注。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335

推荐阅读更多精彩内容

  • 1. 当人们想要并不属于他们的东西的时候,那就是贪婪。 怎么判定一个东西属于不属于一个人呢?如果这个东西属于你,你...
    Inker阅读 189评论 0 0
  • 不管你有多不开心 我们都有责任先吃好一顿饭睡好一个觉打扮好自己 很多烦恼其实都没什么大不了 只是你在那个情境下在那...
    哀慕熙荣阅读 185评论 0 1
  • 今天早上起床后,孩子还是咳嗽,洗漱完后就开始做饭,表哥家的两个孩子也没吃饭,我给三个孩子煮的面条,本以为我家姑娘不...
    小丫_e64d阅读 279评论 0 2