ES6是即将到来的新版本JavaScript语言的标准,代号harmony(和谐之意,显然没有跟上我国的步伐,我们已经进入中国梦版本了)
箭头操作符
如果你会C#或者Java,你肯定知道lambda表达式,ES6中新增的箭头操作符=>便有异曲同工之妙。它简化了函数的书写。操作符左边为输入的参数,而右边则是进行的操作以及返回的值Inputs=>outputs。
我们知道在JS中回调是经常的事,而一般回调又以匿名函数的形式出现,每次都需要写一个function,甚是繁琐。当引入箭头操作符后可以方便地写回调了。请看下面的例子。
var array = [1, 2, 3];
//传统写法
array.forEach(function(v, i, a) { console.log(v);});
//ES6
array.forEach(v = > console.log(v));
类的支持
ES6中添加了对类的支持,引入了class关键字(其实class在JavaScript中一直是保留字,目的就是考虑到可能在以后的新版本中会用到,现在终于派上用场了)。JS本身就是面向对象的,ES6中提供的类实际上只是JS原型模式的包装。现在提供原生的class支持后,对象的创建,继承更加直观了,并且父类方法的调用,实例化,静态方法和构造函数等概念都更加形象化。
//类的定义
class Animal {
//ES6中新型构造器
constructor(name)
{ this.name = name; }
//实例方法
sayName() {
console.log('My name is '+this.name); }}
//类的继承
class Programmer extends Animal { constructor(name) {
//直接调用父类构造器进行初始化
super(name); }
program() { console.log("I'm coding..."); }}
//测试我们的类
var animal=new Animal('dummy'),
wayou=new Programmer('wayou');
animal.sayName();
//输出 ‘My name is dummy’wayou.sayName();
//输出 ‘My name is wayou’wayou.program();
//输出 ‘I'm coding...’
增强的对象字面量
对象字面量被增强了,写法更加简洁与灵活,同时在定义对象的时候能够做的事情更多了。具体表现在:
可以在对象字面量里面定义原型
定义方法可以不用function关键字
直接调用父类方法
这样一来,对象字面量与前面提到的类概念更加吻合,在编写面向对象的JavaScript时更加轻松方便了。
//通过对象字面量创建对象
var human = {
breathe() { console.log('breathing...'); }};
var worker = { __proto__: human,
//设置此对象的原型为human,相当于继承human company: 'freelancer',
work() { console.log('working...'); }};
human.breathe();
//输出 ‘breathing...’
//调用继承来的breathe方法worker.breathe();
//输出 ‘breathing...’
字符串模板
字符串模板相对简单易懂些。
ES6中允许使用反引号 ` 来创建字符串,此种方法创建的字符串里面可以包含由美元符号加花括号包裹的变量${vraible}。如果你使用过像C#等后端强类型语言的话,对此功能应该不会陌生。
//产生一个随机数
var num=Math.random();
//将这个数字输出到
consoleconsole.log(`your num is ${num}`);
解构
自动解析数组或对象中的值。比如若一个函数要返回多个值,常规的做法是返回一个对象,将每个值做为这个对象的属性返回。但在ES6中,利用解构这一特性,可以直接返回一个数组,然后数组中的值会自动被解析到对应接收该值的变量中。
var [x,y]=getVal(),
//函数返回值的解构
[name,,age]=['wayou','male','secrect'];
//数组解构
function getVal()
{ return [ 1, 2 ];}console.log('x:'+x+', y:'+y);
//输出:x:1, y:2 console.log('name:'+name+', age:'+age);
//输出: name:wayou, age:secrect
参数默认值,不定参数,拓展参数
默认参数值
现在可以在定义函数的时候指定参数的默认值了,而不用像以前那样通过逻辑或操作符来达到目的了。
function sayHello(name)
{
//传统的指定默认参数的方式
var name=name||'dude'; console.log('Hello '+name);}
//运用ES6的默认参数
function sayHello2(name='dude'){ console.log(`Hello ${name}`);}
sayHello();
//输出:Hello dudesayHello('Wayou');
//输出:Hello WayousayHello2();
//输出:Hello dudesayHello2('Wayou');
//输出:Hello Wayou
不定参数
不定参数是在函数中使用命名参数同时接收不定数量的未命名参数。这只是一种语法糖,在以前的JavaScript代码中我们可以通过arguments变量来达到这一目的。不定参数的格式是三个句点后跟代表所有不定参数的变量名。比如下面这个例子中,…x代表了所有传入add函数的参数。
//将所有参数相加的函数
function add(...x){ return x.reduce((m,n)=>m+n);}
//传递任意个数的参数console.log(add(1,2,3));
//输出:6console.log(add(1,2,3,4,5));
//输出:15
拓展参数
拓展参数则是另一种形式的语法糖,它允许传递数组或者类数组直接做为函数的参数而不用通过apply。
var people=['Wayou','John','Sherlock'];
//sayHello函数本来接收三个单独的参数人妖,人二和人三
function sayHello(people1,people2,people3){ console.log(`Hello ${people1},${people2},${people3}`);}
//但是我们将一个数组以拓展参数的形式传递,它能很好地映射到每个单独的参数
sayHello(...people);
//输出:Hello Wayou,John,Sherlock
//而在以前,如果需要传递数组当参数,我们需要使用函数的apply方法sayHello.apply(null,people);
//输出:Hello Wayou,John,Sherlock
let与const 关键字
可以把let看成var,只是它定义的变量被限定在了特定范围内才能使用,而离开这个范围则无效。const则很直观,用来定义常量,即无法被更改值的变量。
for (let i=0;i<2;i++)console.log(i);
//输出: 0,1console.log(i);
//输出:undefined,严格模式下会报错
for of 值遍历
我们都知道for in 循环用于遍历数组,类数组或对象,ES6中新引入的for of循环功能相似,不同的是每次循环它提供的不是序号而是值。
var someArray = [ "a", "b", "c" ];
for (v of someArray) { console.log(v);
//输出 a,b,c}
注意,此功能google traceur并未实现,所以无法模拟调试,下面有些功能也是如此
Map,Set 和 WeakMap,WeakSet
这些是新加的集合类型,提供了更加方便的获取属性值的方法,不用像以前一样用hasOwnProperty来检查某个属性是属于原型链上的呢还是当前对象的。同时,在进行属性值添加与获取时有专门的get,set 方法。
Setsvar s = new Set();
s.add("hello").add("goodbye").add("hello");
s.size === 2;
s.has("hello") === true;
Mapsvar m = new Map();
m.set("hello", 42);
m.set(s, 34);
m.get(s) == 34;
有时候我们会把对象作为一个对象的键用来存放属性值,普通集合类型比如简单对象会阻止垃圾回收器对这些作为属性键存在的对象的回收,有造成内存泄漏的危险。而WeakMap,WeakSet则更加安全些,这些作为属性键的对象如果没有别的变量在引用它们,则会被回收释放掉
Math,Number,String,Object 的新API
对Math,Number,String还有Object等添加了许多新的API。下面代码同样来自es6features,对这些新API进行了简单展示。
Number.EPSILONNumber.isInteger(Infinity)
falseNumber.isNaN("NaN")
falseMath.acosh(3) //
1.762747174039086Math.hypot(3, 4) //
5Math.imul(Math.pow(2, 32) - 1, Math.pow(2, 32) - 2) //
2"abcde".contains("cd") //
true"abc".repeat(3) //
"abcabcabc"Array.from(document.querySelectorAll('*')) //
Returns a real ArrayArray.of(1, 2, 3) //
Similar to new Array(...), but without special one-arg behavior[0, 0, 0].fill(7, 1) //
[0,7,7][1,2,3].findIndex(x => x == 2) //
1["a", "b", "c"].entries() //
iterator [0, "a"], [1,"b"], [2,"c"]["a", "b", "c"].keys() //
iterator 0, 1, 2["a", "b", "c"].values() //
iterator "a", "b", "c"Object.assign(Point, { origin: new Point(0,0) })
Promises
Promises是处理异步操作的一种模式,之前在很多三方库中有实现,比如jQuery的deferred 对象。当你发起一个异步请求,并绑定了.when(), .done()等事件处理程序时,其实就是在应用promise模式。
//创建
promisevar promise = new Promise(function(resolve, reject) {
// 进行一些异步或耗时操作
if ( /*如果成功 */ ) { resolve("Stuff worked!"); } else { reject(Error("It broke")); }});
//绑定处理程序
promise.then(function(result) {
//promise成功的话会执行这里 console.log(result);
// "Stuff worked!"
},
function(err) {
//promise失败会执行这里
console.log(err);
// Error: "It broke"});
总结
总结就是一句话,前后端差异越来越小了。