ES6新特性(转)

本文基于lukehoban/es6features ,同时参考了大量博客资料,具体见文末引用。

ES6(ECMAScript 6)是即将到来的新版本JavaScript语言的标准,代号harmony(和谐之意,显然没有跟上我国的步伐,我们已经进入中国梦版本了)。上一次标准的制订还是2009年出台的ES5。目前ES6的标准化工作正在进行中,预计会在14年12月份放出正式敲定的版本。但大部分标准已经就绪,且各浏览器对ES6的支持也正在实现中。要查看ES6的支持情况请点此

目前想要运行ES6代码的话,可以用google/traceur-compiler将代码转译。点此访问traceur-compiler 在线版本时实编辑ES6代码并查看转换后的结果,代码运行结果会在console显示。

另外,关于Google Traceur,业界大神Addy Osmani利用前者写了个Chrome插件ES6 Tepl,安装后也可以进行ES6的测试。

当然,并不是所有ES6新特性都被实现了,所以上面的方法可以测试大部分,有一些还是无法测试的。

虽然ES6都还没真正发布,但已经有用ES6重写的程序了,各种关于ES789的提议已经开始了,这你敢信。潮流不是我等大众所能追赶的。

潮流虽然太快,但我们不停下学习的步伐,就不会被潮流丢下的,下面来领略下ES6中新特性,一堵新生代JS的风采。

箭头操作符

如果你会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));

大家可以打开文章开头提到的traceur在线代码转译页面输入代码来查看效果。

类的支持

ES6中添加了对类的支持,引入了class关键字(其实class在JavaScript中一直是保留字,目的就是考虑到可能在以后的新版本中会用到,现在终于派上用场了)。JS本身就是面向对象的,ES6中提供的类实际上只是JS原型模式的包装。现在提供原生的class支持后,对象的创建,继承更加直观了,并且父类方法的调用,实例化,静态方法和构造函数等概念都更加形象化。

下面代码展示了类在ES6中的使用。再次啰嗦一句,你可以将代码贴到traceur自己查看运行结果。

//类的定义
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();
//将这个数字输出到console
console.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 dude
sayHello('Wayou');//输出:Hello Wayou
sayHello2();//输出:Hello dude
sayHello2('Wayou');//输出:Hello Wayou

 

不定参数

不定参数是在函数中使用命名参数同时接收不定数量的未命名参数。这只是一种语法糖,在以前的JavaScript代码中我们可以通过arguments变量来达到这一目的。不定参数的格式是三个句点后跟代表所有不定参数的变量名。比如下面这个例子中,…x代表了所有传入add函数的参数。

//将所有参数相加的函数
function add(...x){
	return x.reduce((m,n)=>m+n);
}
//传递任意个数的参数
console.log(add(1,2,3));//输出:6
console.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,1
console.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并未实现,所以无法模拟调试,下面有些功能也是如此

iterator, generator

这一部分的内容有点生涩,详情可以参见这里。以下是些基本概念。

  • iterator:它是这么一个对象,拥有一个next方法,这个方法返回一个对象{done,value},这个对象包含两个属性,一个布尔类型的done和包含任意值的value
  • iterable: 这是这么一个对象,拥有一个obj[@@iterator]方法,这个方法返回一个iterator
  • generator: 它是一种特殊的iterator。反的next方法可以接收一个参数并且返回值取决与它的构造函数(generator function)。generator同时拥有一个throw方法
  • generator 函数: 即generator的构造函数。此函数内可以使用yield关键字。在yield出现的地方可以通过generator的nextthrow方法向外界传递值。generator 函数是通过function*来声明的
  • yield 关键字:它可以暂停函数的执行,随后可以再进进入函数继续执行

模块

在ES6标准中,JavaScript原生支持module了。这种将JS代码分割成不同功能的小块进行模块化的概念是在一些三方规范中流行起来的,比如CommonJS和AMD模式。

将不同功能的代码分别写在不同文件中,各模块只需导出公共接口部分,然后通过模块的导入的方式可以在其他地方使用。下面的例子来自tutsplus:

// point.js
module "point" {
    export class Point {
        constructor (x, y) {
            public x = x;
            public y = y;
        }
    }
}
 
// myapp.js
//声明引用的模块
module point from "/point.js";
//这里可以看出,尽管声明了引用的模块,还是可以通过指定需要的部分进行导入
import Point from "point";
 
var origin = new Point(0, 0);
console.log(origin);

Map,Set 和 WeakMap,WeakSet

这些是新加的集合类型,提供了更加方便的获取属性值的方法,不用像以前一样用hasOwnProperty来检查某个属性是属于原型链上的呢还是当前对象的。同时,在进行属性值添加与获取时有专门的getset 方法。

下方代码来自es6feature

// Sets
var s = new Set();
s.add("hello").add("goodbye").add("hello");
s.size === 2;
s.has("hello") === true;

// Maps
var m = new Map();
m.set("hello", 42);
m.set(s, 34);
m.get(s) == 34;

有时候我们会把对象作为一个对象的键用来存放属性值,普通集合类型比如简单对象会阻止垃圾回收器对这些作为属性键存在的对象的回收,有造成内存泄漏的危险。而WeakMap,WeakSet则更加安全些,这些作为属性键的对象如果没有别的变量在引用它们,则会被回收释放掉,具体还看下面的例子。

正文代码来自es6feature

// Weak Maps
var wm = new WeakMap();
wm.set(s, { extra: 42 });
wm.size === undefined

// Weak Sets
var ws = new WeakSet();
ws.add({ data: 42 });//因为添加到ws的这个临时对象没有其他变量引用它,所以ws不会保存它的值,也就是说这次添加其实没有意思

 

Proxies

Proxy可以监听对象身上发生了什么事情,并在这些事情发生后执行一些相应的操作。一下子让我们对一个对象有了很强的追踪能力,同时在数据绑定方面也很有用处。

以下例子借用自这里

//定义被侦听的目标对象
var engineer = { name: 'Joe Sixpack', salary: 50 };
//定义处理程序
var interceptor = {
  set: function (receiver, property, value) {
    console.log(property, 'is changed to', value);
    receiver[property] = value;
  }
};
//创建代理以进行侦听
engineer = Proxy(engineer, interceptor);
//做一些改动来触发代理
engineer.salary = 60;//控制台输出:salary is changed to 60

上面代码我已加了注释,这里进一步解释。对于处理程序,是在被侦听的对象身上发生了相应事件之后,处理程序里面的方法就会被调用,上面例子中我们设置了set的处理函数,表明,如果我们侦听的对象的属性被更改,也就是被set了,那这个处理程序就会被调用,同时通过参数能够得知是哪个属性被更改,更改为了什么值。

Symbols

我们知道对象其实是键值对的集合,而键通常来说是字符串。而现在除了字符串外,我们还可以用symbol这种值来做为对象的键。Symbol是一种基本类型,像数字,字符串还有布尔一样,它不是一个对象。Symbol 通过调用symbol函数产生,它接收一个可选的名字参数,该函数返回的symbol是唯一的。之后就可以用这个返回值做为对象的键了。Symbol还可以用来创建私有属性,外部无法直接访问由symbol做为键的属性值。

以下例子来自es6features

(function() {

  // 创建symbol
  var key = Symbol("key");

  function MyClass(privateData) {
    this[key] = privateData;
  }

  MyClass.prototype = {
    doStuff: function() {
      ... this[key] ...
    }
  };

})();

var c = new MyClass("hello")
c["key"] === undefined//无法访问该属性,因为是私有的

 

Math,Number,String,Object 的新API

对Math,Number,String还有Object等添加了许多新的API。下面代码同样来自es6features,对这些新API进行了简单展示。

Number.EPSILON
Number.isInteger(Infinity) // false
Number.isNaN("NaN") // false

Math.acosh(3) // 1.762747174039086
Math.hypot(3, 4) // 5
Math.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 Array
Array.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模式。

//创建promise
var 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"
});

总结

总结就是一句话,前后端差异越来越小了。

<div id="topics"><div class="post"><h1 class="postTitle"><a id="cb_post_title_url" class="postTitle2" href="http://www.cnblogs.com/Wayou/p/es6_new_features.html">ES6新特性概览</a></h1><div class="clear"></div><div class="postBody"><div id="cnblogs_post_body" class="blogpost-body"><p>本文基于<a href="https://github.com/lukehoban" target="_blank">lukehoban</a>/<a href="https://github.com/lukehoban/es6features" target="_blank"><strong>es6features</strong></a> ,同时参考了大量博客资料,具体见文末引用。</p><p>ES6(<a href="https://www.google.com/url?sa=t&amp;rct=j&amp;q=&amp;esrc=s&amp;source=web&amp;cd=1&amp;cad=rja&amp;uact=8&amp;ved=0CCoQFjAA&amp;url=%68%74%74%70%3a%2f%2f%65%6e%2e%77%69%6b%69%70%65%64%69%61%2e%6f%72%67%2f%77%69%6b%69%2f%45%43%4d%41%53%63%72%69%70%74&amp;ei=Q99JU5jwAoGC8gWq9YDoCQ&amp;usg=AFQjCNGgS-TjEL_-efubqc1RnkUfmn6wew&amp;sig2=Kx2DkQ1Qtv0Wu2E_v_fwGQ&amp;bvm=bv.64542518,d.dGc" target="_blank">ECMAScript</a> 6)是即将到来的新版本JavaScript语言的标准,代号harmony(和谐之意,显然没有跟上我国的步伐,我们已经进入中国梦版本了)。上一次标准的制订还是2009年出台的ES5。目前ES6的标准化工作正在进行中,预计会在14年12月份放出正式敲定的版本。但大部分标准已经就绪,且各浏览器对ES6的支持也正在实现中。要查看ES6的支持情况请<a href="http://kangax.github.io/es5-compat-table/es6/" target="_blank">点此</a>。</p><p>目前想要运行ES6代码的话,可以用<a href="https://github.com/google" target="_blank">google</a>/<a href="https://github.com/google/traceur-compiler" target="_blank"><strong>traceur-compiler</strong></a>将代码转译。<a href="http://google.github.io/traceur-compiler/demo/repl.html" target="_blank">点此</a>访问traceur-compiler 在线版本时实编辑ES6代码并查看转换后的结果,代码运行结果会在console显示。</p><p>另外,关于Google Traceur,业界大神<a href="https://plus.google.com/+AddyOsmani" target="_blank">Addy Osmani</a>利用前者写了个Chrome插件<a href="https://chrome.google.com/webstore/detail/es6-repl/alploljligeomonipppgaahpkenfnfkn" target="_blank">ES6 Tepl</a>,安装后也可以进行ES6的测试。</p><p>当然,并不是所有ES6新特性都被实现了,所以上面的方法可以测试大部分,有一些还是无法测试的。</p><p>虽然ES6都还没真正发布,但已经有用<a href="http://blog.tastejs.com/rewriting-a-webapp-with-ecmascript-6" target="_blank">ES6重写的程序</a>了,各种关于ES789的提议已经开始了,这你敢信。潮流不是我等大众所能追赶的。</p><p>潮流虽然太快,但我们不停下学习的步伐,就不会被潮流丢下的,下面来领略下ES6中新特性,一堵新生代JS的风采。</p><h1>箭头操作符</h1><p>如果你会C#或者Java,你肯定知道lambda表达式,ES6中新增的箭头操作符<span style="background-color: lightgrey;">=&gt;</span>便有异曲同工之妙。它简化了函数的书写。操作符左边为输入的参数,而右边则是进行的操作以及返回的值<span style="background-color: lightgrey;">Inputs=&gt;outputs</span>。</p><p>我们知道在JS中回调是经常的事,而一般回调又以匿名函数的形式出现,每次都需要写一个<span style="background-color: lightgrey;">function</span>,甚是繁琐。当引入箭头操作符后可以方便地写回调了。请看下面的例子。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">array</span> <span style="color: #f92672;">=</span> <span style="color: #f8f8f2;">[</span><span style="color: #ae81ff;">1</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">3</span><span style="color: #f8f8f2;">];</span><span style="color: #75715e;">//传统写法</span><span style="color: #a6e22e;">array</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">forEach</span><span style="color: #f8f8f2;">(</span><span style="color: #66d9ef;">function</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">v</span><span style="color: #f8f8f2;">,</span> <span style="color: #a6e22e;">i</span><span style="color: #f8f8f2;">,</span> <span style="color: #a6e22e;">a</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">v</span><span style="color: #f8f8f2;">);</span><span style="color: #f8f8f2;">});</span><span style="color: #75715e;">//ES6</span><span style="color: #a6e22e;">array</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">forEach</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">v</span> <span style="color: #f92672;">=</span> <span style="color: #f92672;">&gt;</span> <span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">v</span><span style="color: #f8f8f2;">));</span></pre></div><p>大家可以打开文章开头提到的traceur在线代码转译页面输入代码来查看效果。</p><h1>类的支持</h1><p>ES6中添加了对类的支持,引入了<span style="background-color: lightgrey;">class</span>关键字(其实class在JavaScript中一直是保留字,目的就是考虑到可能在以后的新版本中会用到,现在终于派上用场了)。JS本身就是面向对象的,ES6中提供的类实际上只是JS原型模式的包装。现在提供原生的class支持后,对象的创建,继承更加直观了,并且父类方法的调用,实例化,静态方法和构造函数等概念都更加形象化。</p><p>下面代码展示了类在ES6中的使用。再次啰嗦一句,你可以将代码贴到<a href="http://google.github.io/traceur-compiler/demo/repl.html" target="_blank">traceur</a>自己查看运行结果。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #75715e;">//类的定义</span><span style="color: #66d9ef;">class</span> <span style="color: #a6e22e;">Animal</span> <span style="color: #f8f8f2;">{</span><span style="color: #75715e;">//ES6中新型构造器</span>    <span style="color: #a6e22e;">constructor</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">name</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span>        <span style="color: #66d9ef;">this</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">name</span> <span style="color: #f92672;">=</span> <span style="color: #a6e22e;">name</span><span style="color: #f8f8f2;">;</span>    <span style="color: #f8f8f2;">}</span>    <span style="color: #75715e;">//实例方法</span>    <span style="color: #a6e22e;">sayName</span><span style="color: #f8f8f2;">()</span> <span style="color: #f8f8f2;">{</span>        <span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'My name is '</span><span style="color: #f92672;">+</span><span style="color: #66d9ef;">this</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">name</span><span style="color: #f8f8f2;">);</span>    <span style="color: #f8f8f2;">}</span><span style="color: #f8f8f2;">}</span><span style="color: #75715e;">//类的继承</span><span style="color: #66d9ef;">class</span> <span style="color: #a6e22e;">Programmer</span> <span style="color: #66d9ef;">extends</span> <span style="color: #a6e22e;">Animal</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #a6e22e;">constructor</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">name</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #75715e;">//直接调用父类构造器进行初始化</span>        <span style="color: #66d9ef;">super</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">name</span><span style="color: #f8f8f2;">);</span>    <span style="color: #f8f8f2;">}</span>    <span style="color: #a6e22e;">program</span><span style="color: #f8f8f2;">()</span> <span style="color: #f8f8f2;">{</span>        <span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">"I'm coding..."</span><span style="color: #f8f8f2;">);</span>    <span style="color: #f8f8f2;">}</span><span style="color: #f8f8f2;">}</span><span style="color: #75715e;">//测试我们的类</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">animal</span><span style="color: #f92672;">=</span><span style="color: #66d9ef;">new</span> <span style="color: #a6e22e;">Animal</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'dummy'</span><span style="color: #f8f8f2;">),</span><span style="color: #a6e22e;">wayou</span><span style="color: #f92672;">=</span><span style="color: #66d9ef;">new</span> <span style="color: #a6e22e;">Programmer</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'wayou'</span><span style="color: #f8f8f2;">);</span><span style="color: #a6e22e;">animal</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">sayName</span><span style="color: #f8f8f2;">();</span><span style="color: #75715e;">//输出 ‘My name is dummy’</span><span style="color: #a6e22e;">wayou</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">sayName</span><span style="color: #f8f8f2;">();</span><span style="color: #75715e;">//输出 ‘My name is wayou’</span><span style="color: #a6e22e;">wayou</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">program</span><span style="color: #f8f8f2;">();</span><span style="color: #75715e;">//输出 ‘I'm coding...’</span></pre></div><p>&nbsp;</p><h1>增强的对象字面量</h1><p>对象字面量被增强了,写法更加简洁与灵活,同时在定义对象的时候能够做的事情更多了。具体表现在:</p><ul><li>可以在对象字面量里面定义原型</li><li>定义方法可以不用function关键字</li><li>直接调用父类方法</li></ul><p>这样一来,对象字面量与前面提到的类概念更加吻合,在编写面向对象的JavaScript时更加轻松方便了。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #75715e;">//通过对象字面量创建对象</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">human</span> <span style="color: #f92672;">=</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #a6e22e;">breathe</span><span style="color: #f8f8f2;">()</span> <span style="color: #f8f8f2;">{</span>        <span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'breathing...'</span><span style="color: #f8f8f2;">);</span>    <span style="color: #f8f8f2;">}</span><span style="color: #f8f8f2;">};</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">worker</span> <span style="color: #f92672;">=</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #a6e22e;">__proto__</span><span style="color: #f92672;">:</span> <span style="color: #a6e22e;">human</span><span style="color: #f8f8f2;">,</span> <span style="color: #75715e;">//设置此对象的原型为human,相当于继承human</span>    <span style="color: #a6e22e;">company</span><span style="color: #f92672;">:</span> <span style="color: #e6db74;">'freelancer'</span><span style="color: #f8f8f2;">,</span>    <span style="color: #a6e22e;">work</span><span style="color: #f8f8f2;">()</span> <span style="color: #f8f8f2;">{</span>        <span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'working...'</span><span style="color: #f8f8f2;">);</span>    <span style="color: #f8f8f2;">}</span><span style="color: #f8f8f2;">};</span><span style="color: #a6e22e;">human</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">breathe</span><span style="color: #f8f8f2;">();</span><span style="color: #75715e;">//输出 ‘breathing...’</span><span style="color: #75715e;">//调用继承来的breathe方法</span><span style="color: #a6e22e;">worker</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">breathe</span><span style="color: #f8f8f2;">();</span><span style="color: #75715e;">//输出 ‘breathing...’</span></pre></div><p>&nbsp;</p><h1>字符串模板</h1><p>字符串模板相对简单易懂些。ES6中允许使用反引号 <span style="background-color: lightgrey;">`</span> 来创建字符串,此种方法创建的字符串里面可以包含由美元符号加花括号包裹的变量<span style="background-color: lightgrey;">${vraible}</span>。如果你使用过像C#等后端强类型语言的话,对此功能应该不会陌生。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #75715e;">//产生一个随机数</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">num</span><span style="color: #f92672;">=</span><span style="color: #f8f8f2;">Math.</span><span style="color: #a6e22e;">random</span><span style="color: #f8f8f2;">();</span><span style="color: #75715e;">//将这个数字输出到console</span><span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #960050; background-color: #1e0010;">`</span><span style="color: #a6e22e;">your</span> <span style="color: #a6e22e;">num</span> <span style="color: #a6e22e;">is</span> <span style="color: #a6e22e;">$</span><span style="color: #f8f8f2;">{</span><span style="color: #a6e22e;">num</span><span style="color: #f8f8f2;">}</span><span style="color: #960050; background-color: #1e0010;">`</span><span style="color: #f8f8f2;">);</span></pre></div><h1>解构</h1><p>自动解析数组或对象中的值。比如若一个函数要返回多个值,常规的做法是返回一个对象,将每个值做为这个对象的属性返回。但在ES6中,利用解构这一特性,可以直接返回一个数组,然后数组中的值会自动被解析到对应接收该值的变量中。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #66d9ef;">var</span> <span style="color: #f8f8f2;">[</span><span style="color: #a6e22e;">x</span><span style="color: #f8f8f2;">,</span><span style="color: #a6e22e;">y</span><span style="color: #f8f8f2;">]</span><span style="color: #f92672;">=</span><span style="color: #a6e22e;">getVal</span><span style="color: #f8f8f2;">(),</span><span style="color: #75715e;">//函数返回值的解构</span>    <span style="color: #f8f8f2;">[</span><span style="color: #a6e22e;">name</span><span style="color: #f8f8f2;">,,</span><span style="color: #a6e22e;">age</span><span style="color: #f8f8f2;">]</span><span style="color: #f92672;">=</span><span style="color: #f8f8f2;">[</span><span style="color: #e6db74;">'wayou'</span><span style="color: #f8f8f2;">,</span><span style="color: #e6db74;">'male'</span><span style="color: #f8f8f2;">,</span><span style="color: #e6db74;">'secrect'</span><span style="color: #f8f8f2;">];</span><span style="color: #75715e;">//数组解构</span>
<span style="color: #66d9ef;">function</span> <span style="color: #a6e22e;">getVal</span><span style="color: #f8f8f2;">()</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #66d9ef;">return</span> <span style="color: #f8f8f2;">[</span> <span style="color: #ae81ff;">1</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">2</span> <span style="color: #f8f8f2;">];</span><span style="color: #f8f8f2;">}</span>
<span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'x:'</span><span style="color: #f92672;">+</span><span style="color: #a6e22e;">x</span><span style="color: #f92672;">+</span><span style="color: #e6db74;">', y:'</span><span style="color: #f92672;">+</span><span style="color: #a6e22e;">y</span><span style="color: #f8f8f2;">);</span><span style="color: #75715e;">//输出:x:1, y:2 </span><span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'name:'</span><span style="color: #f92672;">+</span><span style="color: #a6e22e;">name</span><span style="color: #f92672;">+</span><span style="color: #e6db74;">', age:'</span><span style="color: #f92672;">+</span><span style="color: #a6e22e;">age</span><span style="color: #f8f8f2;">);</span><span style="color: #75715e;">//输出: name:wayou, age:secrect </span></pre></div><h1>参数默认值,不定参数,拓展参数</h1><h2>默认参数值</h2><p>现在可以在定义函数的时候指定参数的默认值了,而不用像以前那样通过逻辑或操作符来达到目的了。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #66d9ef;">function</span> <span style="color: #a6e22e;">sayHello</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">name</span><span style="color: #f8f8f2;">){</span><span style="color: #75715e;">//传统的指定默认参数的方式</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">name</span><span style="color: #f92672;">=</span><span style="color: #a6e22e;">name</span><span style="color: #f92672;">||</span><span style="color: #e6db74;">'dude'</span><span style="color: #f8f8f2;">;</span><span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'Hello '</span><span style="color: #f92672;">+</span><span style="color: #a6e22e;">name</span><span style="color: #f8f8f2;">);</span><span style="color: #f8f8f2;">}</span><span style="color: #75715e;">//运用ES6的默认参数</span><span style="color: #66d9ef;">function</span> <span style="color: #a6e22e;">sayHello2</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">name</span><span style="color: #f92672;">=</span><span style="color: #e6db74;">'dude'</span><span style="color: #f8f8f2;">){</span><span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #960050; background-color: #1e0010;">`</span><span style="color: #a6e22e;">Hello</span> <span style="color: #a6e22e;">$</span><span style="color: #f8f8f2;">{</span><span style="color: #a6e22e;">name</span><span style="color: #f8f8f2;">}</span><span style="color: #960050; background-color: #1e0010;">`</span><span style="color: #f8f8f2;">);</span><span style="color: #f8f8f2;">}</span><span style="color: #a6e22e;">sayHello</span><span style="color: #f8f8f2;">();</span><span style="color: #75715e;">//输出:Hello dude</span><span style="color: #a6e22e;">sayHello</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'Wayou'</span><span style="color: #f8f8f2;">);</span><span style="color: #75715e;">//输出:Hello Wayou</span><span style="color: #a6e22e;">sayHello2</span><span style="color: #f8f8f2;">();</span><span style="color: #75715e;">//输出:Hello dude</span><span style="color: #a6e22e;">sayHello2</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'Wayou'</span><span style="color: #f8f8f2;">);</span><span style="color: #75715e;">//输出:Hello Wayou</span></pre></div><p>&nbsp;</p><h2>不定参数</h2><p>不定参数是在函数中使用命名参数同时接收不定数量的未命名参数。这只是一种语法糖,在以前的JavaScript代码中我们可以通过arguments变量来达到这一目的。不定参数的格式是三个句点后跟代表所有不定参数的变量名。比如下面这个例子中,<span style="background-color: lightgrey;">…x</span>代表了所有传入<span style="background-color: lightgrey;">add</span>函数的参数。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #75715e;">//将所有参数相加的函数</span><span style="color: #66d9ef;">function</span> <span style="color: #a6e22e;">add</span><span style="color: #f8f8f2;">(...</span><span style="color: #a6e22e;">x</span><span style="color: #f8f8f2;">){</span><span style="color: #66d9ef;">return</span> <span style="color: #a6e22e;">x</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">reduce</span><span style="color: #f8f8f2;">((</span><span style="color: #a6e22e;">m</span><span style="color: #f8f8f2;">,</span><span style="color: #a6e22e;">n</span><span style="color: #f8f8f2;">)</span><span style="color: #f92672;">=&gt;</span><span style="color: #a6e22e;">m</span><span style="color: #f92672;">+</span><span style="color: #a6e22e;">n</span><span style="color: #f8f8f2;">);</span><span style="color: #f8f8f2;">}</span><span style="color: #75715e;">//传递任意个数的参数</span><span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">add</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">1</span><span style="color: #f8f8f2;">,</span><span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">,</span><span style="color: #ae81ff;">3</span><span style="color: #f8f8f2;">));</span><span style="color: #75715e;">//输出:6</span><span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">add</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">1</span><span style="color: #f8f8f2;">,</span><span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">,</span><span style="color: #ae81ff;">3</span><span style="color: #f8f8f2;">,</span><span style="color: #ae81ff;">4</span><span style="color: #f8f8f2;">,</span><span style="color: #ae81ff;">5</span><span style="color: #f8f8f2;">));</span><span style="color: #75715e;">//输出:15</span></pre></div><p>&nbsp;</p><h2>拓展参数</h2><p>拓展参数则是另一种形式的语法糖,它允许传递数组或者类数组直接做为函数的参数而不用通过<span style="background-color: lightgrey;">apply</span>。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">people</span><span style="color: #f92672;">=</span><span style="color: #f8f8f2;">[</span><span style="color: #e6db74;">'Wayou'</span><span style="color: #f8f8f2;">,</span><span style="color: #e6db74;">'John'</span><span style="color: #f8f8f2;">,</span><span style="color: #e6db74;">'Sherlock'</span><span style="color: #f8f8f2;">];</span><span style="color: #75715e;">//sayHello函数本来接收三个单独的参数人妖,人二和人三</span><span style="color: #66d9ef;">function</span> <span style="color: #a6e22e;">sayHello</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">people1</span><span style="color: #f8f8f2;">,</span><span style="color: #a6e22e;">people2</span><span style="color: #f8f8f2;">,</span><span style="color: #a6e22e;">people3</span><span style="color: #f8f8f2;">){</span><span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #960050; background-color: #1e0010;">`</span><span style="color: #a6e22e;">Hello</span> <span style="color: #a6e22e;">$</span><span style="color: #f8f8f2;">{</span><span style="color: #a6e22e;">people1</span><span style="color: #f8f8f2;">},</span><span style="color: #a6e22e;">$</span><span style="color: #f8f8f2;">{</span><span style="color: #a6e22e;">people2</span><span style="color: #f8f8f2;">},</span><span style="color: #a6e22e;">$</span><span style="color: #f8f8f2;">{</span><span style="color: #a6e22e;">people3</span><span style="color: #f8f8f2;">}</span><span style="color: #960050; background-color: #1e0010;">`</span><span style="color: #f8f8f2;">);</span><span style="color: #f8f8f2;">}</span><span style="color: #75715e;">//但是我们将一个数组以拓展参数的形式传递,它能很好地映射到每个单独的参数</span><span style="color: #a6e22e;">sayHello</span><span style="color: #f8f8f2;">(...</span><span style="color: #a6e22e;">people</span><span style="color: #f8f8f2;">);</span><span style="color: #75715e;">//输出:Hello Wayou,John,Sherlock </span>
<span style="color: #75715e;">//而在以前,如果需要传递数组当参数,我们需要使用函数的apply方法</span><span style="color: #a6e22e;">sayHello</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">apply</span><span style="color: #f8f8f2;">(</span><span style="color: #66d9ef;">null</span><span style="color: #f8f8f2;">,</span><span style="color: #a6e22e;">people</span><span style="color: #f8f8f2;">);</span><span style="color: #75715e;">//输出:Hello Wayou,John,Sherlock </span></pre></div><h1>let与const 关键字</h1><p>可以把<span style="background-color: lightgrey;">let</span>看成<span style="background-color: lightgrey;">var</span>,只是它定义的变量被限定在了特定范围内才能使用,而离开这个范围则无效。<span style="background-color: lightgrey;">const</span>则很直观,用来定义常量,即无法被更改值的变量。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #66d9ef;">for</span> <span style="color: #f8f8f2;">(</span><span style="color: #66d9ef;">let</span> <span style="color: #a6e22e;">i</span><span style="color: #f92672;">=</span><span style="color: #ae81ff;">0</span><span style="color: #f8f8f2;">;</span><span style="color: #a6e22e;">i</span><span style="color: #f92672;">&lt;</span><span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">;</span><span style="color: #a6e22e;">i</span><span style="color: #f92672;">++</span><span style="color: #f8f8f2;">)</span><span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">i</span><span style="color: #f8f8f2;">);</span><span style="color: #75715e;">//输出: 0,1</span><span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">i</span><span style="color: #f8f8f2;">);</span><span style="color: #75715e;">//输出:undefined,严格模式下会报错</span></pre></div><h1>for of 值遍历</h1><p>我们都知道<span style="background-color: lightgrey;">for in</span> 循环用于遍历数组,类数组或对象,ES6中新引入的<span style="background-color: lightgrey;">for of</span>循环功能相似,不同的是每次循环它提供的不是序号而是值。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">someArray</span> <span style="color: #f92672;">=</span> <span style="color: #f8f8f2;">[</span> <span style="color: #e6db74;">"a"</span><span style="color: #f8f8f2;">,</span> <span style="color: #e6db74;">"b"</span><span style="color: #f8f8f2;">,</span> <span style="color: #e6db74;">"c"</span> <span style="color: #f8f8f2;">];</span> <span style="color: #66d9ef;">for</span> <span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">v</span> <span style="color: #a6e22e;">of</span> <span style="color: #a6e22e;">someArray</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">v</span><span style="color: #f8f8f2;">);</span><span style="color: #75715e;">//输出 a,b,c</span><span style="color: #f8f8f2;">}</span></pre></div><p class="emphasis_block"><em>注意,此功能google traceur并未实现,所以无法模拟调试,下面有些功能也是如此</em></p><h1>iterator, generator</h1><p>这一部分的内容有点生涩,详情可以参见<a href="http://domenic.me/2013/09/06/es6-iterators-generators-and-iterables/" target="_blank">这里</a>。以下是些基本概念。</p><ul><li>iterator:它是这么一个对象,拥有一个next方法,这个方法返回一个对象{done,value},这个对象包含两个属性,一个布尔类型的done和包含任意值的value</li><li>iterable: 这是这么一个对象,拥有一个obj[@@iterator]方法,这个方法返回一个iterator</li><li>generator: 它是一种特殊的iterator。反的next方法可以接收一个参数并且返回值取决与它的构造函数(generator function)。generator同时拥有一个throw方法</li><li>generator 函数: 即generator的构造函数。此函数内可以使用<span style="background-color: lightgrey;">yield</span>关键字。在<span style="background-color: lightgrey;">yield</span>出现的地方可以通过generator的<span style="background-color: lightgrey;">next</span>或<span style="background-color: lightgrey;">throw</span>方法向外界传递值。generator 函数是通过<span style="background-color: lightgrey;">function*</span>来声明的</li><li>yield 关键字:它可以暂停函数的执行,随后可以再进进入函数继续执行</li></ul><h1>模块</h1><p>在ES6标准中,JavaScript原生支持module了。这种将JS代码分割成不同功能的小块进行模块化的概念是在一些三方规范中流行起来的,比如CommonJS和AMD模式。</p><p>将不同功能的代码分别写在不同文件中,各模块只需导出公共接口部分,然后通过模块的导入的方式可以在其他地方使用。下面的例子来自<a href="http://code.tutsplus.com/tutorials/eight-cool-features-coming-in-es6--net-33175" target="_blank">tutsplus</a>:</p><div class="highlight" style="background: #272822;"><pre><span style="color: #75715e;">// point.js</span><span style="color: #a6e22e;">module</span> <span style="color: #e6db74;">"point"</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #66d9ef;">export</span> <span style="color: #66d9ef;">class</span> <span style="color: #a6e22e;">Point</span> <span style="color: #f8f8f2;">{</span>        <span style="color: #a6e22e;">constructor</span> <span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">x</span><span style="color: #f8f8f2;">,</span> <span style="color: #a6e22e;">y</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span>            <span style="color: #66d9ef;">public</span> <span style="color: #a6e22e;">x</span> <span style="color: #f92672;">=</span> <span style="color: #a6e22e;">x</span><span style="color: #f8f8f2;">;</span>            <span style="color: #66d9ef;">public</span> <span style="color: #a6e22e;">y</span> <span style="color: #f92672;">=</span> <span style="color: #a6e22e;">y</span><span style="color: #f8f8f2;">;</span>        <span style="color: #f8f8f2;">}</span>    <span style="color: #f8f8f2;">}</span><span style="color: #f8f8f2;">}</span> <span style="color: #75715e;">// myapp.js</span><span style="color: #75715e;">//声明引用的模块</span><span style="color: #a6e22e;">module</span> <span style="color: #a6e22e;">point</span> <span style="color: #a6e22e;">from</span> <span style="color: #e6db74;">"/point.js"</span><span style="color: #f8f8f2;">;</span><span style="color: #75715e;">//这里可以看出,尽管声明了引用的模块,还是可以通过指定需要的部分进行导入</span><span style="color: #66d9ef;">import</span> <span style="color: #a6e22e;">Point</span> <span style="color: #a6e22e;">from</span> <span style="color: #e6db74;">"point"</span><span style="color: #f8f8f2;">;</span> <span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">origin</span> <span style="color: #f92672;">=</span> <span style="color: #66d9ef;">new</span> <span style="color: #a6e22e;">Point</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">0</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">0</span><span style="color: #f8f8f2;">);</span><span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">origin</span><span style="color: #f8f8f2;">);</span></pre></div><h1>Map,Set 和 WeakMap,WeakSet</h1><p>这些是新加的集合类型,提供了更加方便的获取属性值的方法,不用像以前一样用<span style="background-color: lightgrey;">hasOwnProperty</span>来检查某个属性是属于原型链上的呢还是当前对象的。同时,在进行属性值添加与获取时有专门的<span style="background-color: lightgrey;">get</span>,<span style="background-color: lightgrey;">set</span> 方法。</p><p>下方代码来自<a href="https://github.com/lukehoban/es6features" target="_blank">es6feature</a></p><div class="highlight" style="background: #272822;"><pre><span style="color: #75715e;">// Sets</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">s</span> <span style="color: #f92672;">=</span> <span style="color: #66d9ef;">new</span> <span style="color: #a6e22e;">Set</span><span style="color: #f8f8f2;">();</span><span style="color: #a6e22e;">s</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">add</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">"hello"</span><span style="color: #f8f8f2;">).</span><span style="color: #a6e22e;">add</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">"goodbye"</span><span style="color: #f8f8f2;">).</span><span style="color: #a6e22e;">add</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">"hello"</span><span style="color: #f8f8f2;">);</span><span style="color: #a6e22e;">s</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">size</span> <span style="color: #f92672;">===</span> <span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">;</span><span style="color: #a6e22e;">s</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">has</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">"hello"</span><span style="color: #f8f8f2;">)</span> <span style="color: #f92672;">===</span> <span style="color: #66d9ef;">true</span><span style="color: #f8f8f2;">;</span>
<span style="color: #75715e;">// Maps</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">m</span> <span style="color: #f92672;">=</span> <span style="color: #66d9ef;">new</span> <span style="color: #a6e22e;">Map</span><span style="color: #f8f8f2;">();</span><span style="color: #a6e22e;">m</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">set</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">"hello"</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">42</span><span style="color: #f8f8f2;">);</span><span style="color: #a6e22e;">m</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">set</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">s</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">34</span><span style="color: #f8f8f2;">);</span><span style="color: #a6e22e;">m</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">get</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">s</span><span style="color: #f8f8f2;">)</span> <span style="color: #f92672;">==</span> <span style="color: #ae81ff;">34</span><span style="color: #f8f8f2;">;</span></pre></div><p>有时候我们会把对象作为一个对象的键用来存放属性值,普通集合类型比如简单对象会阻止垃圾回收器对这些作为属性键存在的对象的回收,有造成内存泄漏的危险。而<span style="background-color: #c0c0c0;">WeakMap</span>,<span style="background-color: #c0c0c0;">WeakSet</span>则更加安全些,这些作为属性键的对象如果没有别的变量在引用它们,则会被回收释放掉,具体还看下面的例子。</p><p>正文代码来自<a href="https://github.com/lukehoban/es6features" target="_blank">es6feature</a></p><div class="highlight" style="background: #272822;"><pre><span style="color: #75715e;">// Weak Maps</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">wm</span> <span style="color: #f92672;">=</span> <span style="color: #66d9ef;">new</span> <span style="color: #a6e22e;">WeakMap</span><span style="color: #f8f8f2;">();</span><span style="color: #a6e22e;">wm</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">set</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">s</span><span style="color: #f8f8f2;">,</span> <span style="color: #f8f8f2;">{</span> <span style="color: #a6e22e;">extra</span><span style="color: #f92672;">:</span> <span style="color: #ae81ff;">42</span> <span style="color: #f8f8f2;">});</span><span style="color: #a6e22e;">wm</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">size</span> <span style="color: #f92672;">===</span> <span style="color: #66d9ef;">undefined</span>
<span style="color: #75715e;">// Weak Sets</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">ws</span> <span style="color: #f92672;">=</span> <span style="color: #66d9ef;">new</span> <span style="color: #a6e22e;">WeakSet</span><span style="color: #f8f8f2;">();</span><span style="color: #a6e22e;">ws</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">add</span><span style="color: #f8f8f2;">({</span> <span style="color: #a6e22e;">data</span><span style="color: #f92672;">:</span> <span style="color: #ae81ff;">42</span> <span style="color: #f8f8f2;">});</span><span style="color: #75715e;">//因为添加到ws的这个临时对象没有其他变量引用它,所以ws不会保存它的值,也就是说这次添加其实没有意思</span></pre></div><p>&nbsp;</p><h1>Proxies</h1><p>Proxy可以监听对象身上发生了什么事情,并在这些事情发生后执行一些相应的操作。一下子让我们对一个对象有了很强的追踪能力,同时在数据绑定方面也很有用处。</p><p>以下例子借用自<a href="http://ariya.ofilabs.com/2013/07/es6-and-proxy.html" target="_blank">这里</a>。<code><span style="color: #75715e;"><span style="color: #f8f8f2; font-family: 'Courier New'; line-height: 1.5;"><br></span></span></code></p><div class="highlight" style="background: #272822;"><pre><span style="color: #75715e;">//定义被侦听的目标对象</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">engineer</span> <span style="color: #f92672;">=</span> <span style="color: #f8f8f2;">{</span> <span style="color: #a6e22e;">name</span><span style="color: #f92672;">:</span> <span style="color: #e6db74;">'Joe Sixpack'</span><span style="color: #f8f8f2;">,</span> <span style="color: #a6e22e;">salary</span><span style="color: #f92672;">:</span> <span style="color: #ae81ff;">50</span> <span style="color: #f8f8f2;">};</span><span style="color: #75715e;">//定义处理程序</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">interceptor</span> <span style="color: #f92672;">=</span> <span style="color: #f8f8f2;">{</span>  <span style="color: #a6e22e;">set</span><span style="color: #f92672;">:</span> <span style="color: #66d9ef;">function</span> <span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">receiver</span><span style="color: #f8f8f2;">,</span> <span style="color: #a6e22e;">property</span><span style="color: #f8f8f2;">,</span> <span style="color: #a6e22e;">value</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">property</span><span style="color: #f8f8f2;">,</span> <span style="color: #e6db74;">'is changed to'</span><span style="color: #f8f8f2;">,</span> <span style="color: #a6e22e;">value</span><span style="color: #f8f8f2;">);</span>    <span style="color: #a6e22e;">receiver</span><span style="color: #f8f8f2;">[</span><span style="color: #a6e22e;">property</span><span style="color: #f8f8f2;">]</span> <span style="color: #f92672;">=</span> <span style="color: #a6e22e;">value</span><span style="color: #f8f8f2;">;</span>  <span style="color: #f8f8f2;">}</span><span style="color: #f8f8f2;">};</span><span style="color: #75715e;">//创建代理以进行侦听</span><span style="color: #a6e22e;">engineer</span> <span style="color: #f92672;">=</span> <span style="color: #a6e22e;">Proxy</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">engineer</span><span style="color: #f8f8f2;">,</span> <span style="color: #a6e22e;">interceptor</span><span style="color: #f8f8f2;">);</span><span style="color: #75715e;">//做一些改动来触发代理</span><span style="color: #a6e22e;">engineer</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">salary</span> <span style="color: #f92672;">=</span> <span style="color: #ae81ff;">60</span><span style="color: #f8f8f2;">;</span><span style="color: #75715e;">//控制台输出:salary is changed to 60</span></pre></div><p>上面代码我已加了注释,这里进一步解释。对于处理程序,是在被侦听的对象身上发生了相应事件之后,处理程序里面的方法就会被调用,上面例子中我们设置了<span style="background-color: lightgrey;">set</span>的处理函数,表明,如果我们侦听的对象的属性被更改,也就是被<span style="background-color: lightgrey;">set</span>了,那这个处理程序就会被调用,同时通过参数能够得知是哪个属性被更改,更改为了什么值。</p><h1>Symbols</h1><p>我们知道对象其实是键值对的集合,而键通常来说是字符串。而现在除了字符串外,我们还可以用symbol这种值来做为对象的键。Symbol是一种基本类型,像数字,字符串还有布尔一样,它不是一个对象。Symbol 通过调用symbol函数产生,它接收一个可选的名字参数,该函数返回的symbol是唯一的。之后就可以用这个返回值做为对象的键了。Symbol还可以用来创建私有属性,外部无法直接访问由symbol做为键的属性值。</p><p>以下例子来自<a href="https://github.com/lukehoban/es6features" target="_blank">es6features</a></p><div class="highlight" style="background: #272822;"><pre><span style="color: #f8f8f2;">(</span><span style="color: #66d9ef;">function</span><span style="color: #f8f8f2;">()</span> <span style="color: #f8f8f2;">{</span>
  <span style="color: #75715e;">// 创建symbol</span>  <span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">key</span> <span style="color: #f92672;">=</span> <span style="color: #a6e22e;">Symbol</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">"key"</span><span style="color: #f8f8f2;">);</span>
  <span style="color: #66d9ef;">function</span> <span style="color: #a6e22e;">MyClass</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">privateData</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #66d9ef;">this</span><span style="color: #f8f8f2;">[</span><span style="color: #a6e22e;">key</span><span style="color: #f8f8f2;">]</span> <span style="color: #f92672;">=</span> <span style="color: #a6e22e;">privateData</span><span style="color: #f8f8f2;">;</span>  <span style="color: #f8f8f2;">}</span>
  <span style="color: #a6e22e;">MyClass</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">prototype</span> <span style="color: #f92672;">=</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #a6e22e;">doStuff</span><span style="color: #f92672;">:</span> <span style="color: #66d9ef;">function</span><span style="color: #f8f8f2;">()</span> <span style="color: #f8f8f2;">{</span>      <span style="color: #f8f8f2;">...</span> <span style="color: #66d9ef;">this</span><span style="color: #f8f8f2;">[</span><span style="color: #a6e22e;">key</span><span style="color: #f8f8f2;">]</span> <span style="color: #f8f8f2;">...</span>    <span style="color: #f8f8f2;">}</span>  <span style="color: #f8f8f2;">};</span>
<span style="color: #f8f8f2;">})();</span>
<span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">c</span> <span style="color: #f92672;">=</span> <span style="color: #66d9ef;">new</span> <span style="color: #a6e22e;">MyClass</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">"hello"</span><span style="color: #f8f8f2;">)</span><span style="color: #a6e22e;">c</span><span style="color: #f8f8f2;">[</span><span style="color: #e6db74;">"key"</span><span style="color: #f8f8f2;">]</span> <span style="color: #f92672;">===</span> <span style="color: #66d9ef;">undefined</span><span style="color: #75715e;">//无法访问该属性,因为是私有的</span></pre></div><p>&nbsp;</p><h1>Math,Number,String,Object 的新API</h1><p>对Math,Number,String还有Object等添加了许多新的API。下面代码同样来自es6features,对这些新API进行了简单展示。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #f8f8f2;">Number.</span><span style="color: #a6e22e;">EPSILON</span><span style="color: #f8f8f2;">Number.</span><span style="color: #a6e22e;">isInteger</span><span style="color: #f8f8f2;">(</span><span style="color: #66d9ef;">Infinity</span><span style="color: #f8f8f2;">)</span> <span style="color: #75715e;">// false</span><span style="color: #f8f8f2;">Number.isNaN(</span><span style="color: #e6db74;">"NaN"</span><span style="color: #f8f8f2;">)</span> <span style="color: #75715e;">// false</span>
<span style="color: #f8f8f2;">Math.</span><span style="color: #a6e22e;">acosh</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">3</span><span style="color: #f8f8f2;">)</span> <span style="color: #75715e;">// 1.762747174039086</span><span style="color: #f8f8f2;">Math.</span><span style="color: #a6e22e;">hypot</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">3</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">4</span><span style="color: #f8f8f2;">)</span> <span style="color: #75715e;">// 5</span><span style="color: #f8f8f2;">Math.</span><span style="color: #a6e22e;">imul</span><span style="color: #f8f8f2;">(Math.</span><span style="color: #a6e22e;">pow</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">32</span><span style="color: #f8f8f2;">)</span> <span style="color: #f92672;">-</span> <span style="color: #ae81ff;">1</span><span style="color: #f8f8f2;">,</span> <span style="color: #f8f8f2;">Math.</span><span style="color: #a6e22e;">pow</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">32</span><span style="color: #f8f8f2;">)</span> <span style="color: #f92672;">-</span> <span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">)</span> <span style="color: #75715e;">// 2</span>
<span style="color: #e6db74;">"abcde"</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">contains</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">"cd"</span><span style="color: #f8f8f2;">)</span> <span style="color: #75715e;">// true</span><span style="color: #e6db74;">"abc"</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">repeat</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">3</span><span style="color: #f8f8f2;">)</span> <span style="color: #75715e;">// "abcabcabc"</span>
<span style="color: #f8f8f2;">Array.</span><span style="color: #a6e22e;">from</span><span style="color: #f8f8f2;">(document.</span><span style="color: #a6e22e;">querySelectorAll</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">'*'</span><span style="color: #f8f8f2;">))</span> <span style="color: #75715e;">// Returns a real Array</span><span style="color: #f8f8f2;">Array.</span><span style="color: #a6e22e;">of</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">1</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">3</span><span style="color: #f8f8f2;">)</span> <span style="color: #75715e;">// Similar to new Array(...), but without special one-arg behavior</span><span style="color: #f8f8f2;">[</span><span style="color: #ae81ff;">0</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">0</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">0</span><span style="color: #f8f8f2;">].</span><span style="color: #a6e22e;">fill</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">7</span><span style="color: #f8f8f2;">,</span> <span style="color: #ae81ff;">1</span><span style="color: #f8f8f2;">)</span> <span style="color: #75715e;">// [0,7,7]</span><span style="color: #f8f8f2;">[</span><span style="color: #ae81ff;">1</span><span style="color: #f8f8f2;">,</span><span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">,</span><span style="color: #ae81ff;">3</span><span style="color: #f8f8f2;">].</span><span style="color: #a6e22e;">findIndex</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">x</span> <span style="color: #f92672;">=&gt;</span> <span style="color: #a6e22e;">x</span> <span style="color: #f92672;">==</span> <span style="color: #ae81ff;">2</span><span style="color: #f8f8f2;">)</span> <span style="color: #75715e;">// 1</span><span style="color: #f8f8f2;">[</span><span style="color: #e6db74;">"a"</span><span style="color: #f8f8f2;">,</span> <span style="color: #e6db74;">"b"</span><span style="color: #f8f8f2;">,</span> <span style="color: #e6db74;">"c"</span><span style="color: #f8f8f2;">].</span><span style="color: #a6e22e;">entries</span><span style="color: #f8f8f2;">()</span> <span style="color: #75715e;">// iterator [0, "a"], [1,"b"], [2,"c"]</span><span style="color: #f8f8f2;">[</span><span style="color: #e6db74;">"a"</span><span style="color: #f8f8f2;">,</span> <span style="color: #e6db74;">"b"</span><span style="color: #f8f8f2;">,</span> <span style="color: #e6db74;">"c"</span><span style="color: #f8f8f2;">].</span><span style="color: #a6e22e;">keys</span><span style="color: #f8f8f2;">()</span> <span style="color: #75715e;">// iterator 0, 1, 2</span><span style="color: #f8f8f2;">[</span><span style="color: #e6db74;">"a"</span><span style="color: #f8f8f2;">,</span> <span style="color: #e6db74;">"b"</span><span style="color: #f8f8f2;">,</span> <span style="color: #e6db74;">"c"</span><span style="color: #f8f8f2;">].</span><span style="color: #a6e22e;">values</span><span style="color: #f8f8f2;">()</span> <span style="color: #75715e;">// iterator "a", "b", "c"</span>
<span style="color: #f8f8f2;">Object.</span><span style="color: #a6e22e;">assign</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">Point</span><span style="color: #f8f8f2;">,</span> <span style="color: #f8f8f2;">{</span> <span style="color: #a6e22e;">origin</span><span style="color: #f92672;">:</span> <span style="color: #66d9ef;">new</span> <span style="color: #a6e22e;">Point</span><span style="color: #f8f8f2;">(</span><span style="color: #ae81ff;">0</span><span style="color: #f8f8f2;">,</span><span style="color: #ae81ff;">0</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">})</span></pre></div><p>&nbsp;</p><h1>Promises</h1><p>Promises是处理异步操作的一种模式,之前在很多三方库中有实现,比如jQuery的<a href="http://api.jquery.com/category/deferred-object/" target="_blank">deferred</a> 对象。当你发起一个异步请求,并绑定了<span style="background-color: silver;">.when()</span>, <span style="background-color: silver;">.done()</span>等事件处理程序时,其实就是在应用promise模式。</p><div class="highlight" style="background: #272822;"><pre><span style="color: #75715e;">//创建promise</span><span style="color: #66d9ef;">var</span> <span style="color: #a6e22e;">promise</span> <span style="color: #f92672;">=</span> <span style="color: #66d9ef;">new</span> <span style="color: #a6e22e;">Promise</span><span style="color: #f8f8f2;">(</span><span style="color: #66d9ef;">function</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">resolve</span><span style="color: #f8f8f2;">,</span> <span style="color: #a6e22e;">reject</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span>    <span style="color: #75715e;">// 进行一些异步或耗时操作</span>    <span style="color: #66d9ef;">if</span> <span style="color: #f8f8f2;">(</span> <span style="color: #75715e;">/*如果成功 */</span> <span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span>        <span style="color: #a6e22e;">resolve</span><span style="color: #f8f8f2;">(</span><span style="color: #e6db74;">"Stuff worked!"</span><span style="color: #f8f8f2;">);</span>    <span style="color: #f8f8f2;">}</span> <span style="color: #66d9ef;">else</span> <span style="color: #f8f8f2;">{</span>        <span style="color: #a6e22e;">reject</span><span style="color: #f8f8f2;">(Error(</span><span style="color: #e6db74;">"It broke"</span><span style="color: #f8f8f2;">));</span>    <span style="color: #f8f8f2;">}</span><span style="color: #f8f8f2;">});</span><span style="color: #75715e;">//绑定处理程序</span><span style="color: #a6e22e;">promise</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">then</span><span style="color: #f8f8f2;">(</span><span style="color: #66d9ef;">function</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">result</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span><span style="color: #75715e;">//promise成功的话会执行这里</span>    <span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">result</span><span style="color: #f8f8f2;">);</span> <span style="color: #75715e;">// "Stuff worked!"</span><span style="color: #f8f8f2;">},</span> <span style="color: #66d9ef;">function</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">err</span><span style="color: #f8f8f2;">)</span> <span style="color: #f8f8f2;">{</span><span style="color: #75715e;">//promise失败会执行这里</span>    <span style="color: #a6e22e;">console</span><span style="color: #f8f8f2;">.</span><span style="color: #a6e22e;">log</span><span style="color: #f8f8f2;">(</span><span style="color: #a6e22e;">err</span><span style="color: #f8f8f2;">);</span> <span style="color: #75715e;">// Error: "It broke"</span><span style="color: #f8f8f2;">});</span></pre></div><h1>总结</h1><p>总结就是一句话,前后端差异越来越小了。</p><h1>REFERENCE</h1><ul><li>Google traceur online compiler <a href="http://google.github.io/traceur-compiler/demo/repl.html" target="_blank">http://google.github.io/traceur-compiler/demo/repl.html</a></li><li>array destruction <a href="http://ariya.ofilabs.com/2013/02/es6-and-destructuring-assignment.html" target="_blank">http://ariya.ofilabs.com/2013/02/es6-and-destructuring-assignment.html</a></li><li>class <a href="http://www.joezimjs.com/javascript/javascript-prototypal-inheritance-and-what-es6-classes-have-to-say-about-it/" target="_blank">http://www.joezimjs.com/javascript/javascript-prototypal-inheritance-and-what-es6-classes-have-to-say-about-it/</a></li><li>enhanced object literal <a href="http://maximilianhoffmann.com/posts/object-based-javascript-in-es6" target="_blank">http://maximilianhoffmann.com/posts/object-based-javascript-in-es6</a></li><li>parameters <a href="http://globaldev.co.uk/2013/10/es6-part-4/" target="_blank">http://globaldev.co.uk/2013/10/es6-part-4/</a></li><li>let keyword <a href="http://globaldev.co.uk/2013/09/es6-part-2/" target="_blank">http://globaldev.co.uk/2013/09/es6-part-2/</a></li><li>for of iterator <a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/for...of" target="_blank">https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/for...of#Browser_compatibility</a></li><li>the Iterator protocol <a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/The_Iterator_protocol" target="_blank">https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/The_Iterator_protocol</a></li><li>generators <a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/function*" target="_blank">https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/function*</a></li><li>ES6 Iterators, Generators, and Iterables <a href="http://domenic.me/2013/09/06/es6-iterators-generators-and-iterables/" target="_blank">http://domenic.me/2013/09/06/es6-iterators-generators-and-iterables/</a></li><li>proxies <a href="http://ariya.ofilabs.com/2013/07/es6-and-proxy.html" target="_blank">http://ariya.ofilabs.com/2013/07/es6-and-proxy.html</a></li><li>symbols <a href="http://tc39wiki.calculist.org/es6/symbols/" target="_blank">http://tc39wiki.calculist.org/es6/symbols/</a></li><li>promise http://www.html5rocks.com/en/tutorials/es6/promises/</li><li>8 cool features in ES6 <a href="http://code.tutsplus.com/tutorials/eight-cool-features-coming-in-es6--net-33175" target="_blank">http://code.tutsplus.com/tutorials/eight-cool-features-coming-in-es6--net-33175</a> (此文章错误较多)</li></ul><a class="fastCom" id="fastCom" title="快速评论" href="#comment_form_container"></a></div><div id="MySignature" style="display: block;"><p>Feel free to repost but keep the link to this page please!</p></div><div class="clear"></div><div id="blog_post_info_block"><div id="BlogPostCategory">分类: <a href="http://www.cnblogs.com/Wayou/category/467062.html" target="_blank">JavaScript</a></div><div id="EntryTag">标签: <a href="http://www.cnblogs.com/Wayou/tag/javascript/">javascript</a>, <a href="http://www.cnblogs.com/Wayou/tag/es6/">es6</a>, <a href="http://www.cnblogs.com/Wayou/tag/ecmascript6/">ecmascript6</a>, <a href="http://www.cnblogs.com/Wayou/tag/harmony/">harmony</a></div><div id="blog_post_info"><div id="green_channel">        <a href="javascript:void(0);" id="green_channel_digg" onclick="DiggIt(3663567,cb_blogId,1);green_channel_success(this,'谢谢推荐!');">好文要顶</a>            <a id="green_channel_follow" onclick="follow('c5e8573b-1ed6-e111-aa3f-842b2b196315');" href="javascript:void(0);">关注我</a>    <a id="green_channel_favorite" onclick="AddToWz(cb_entryId);return false;" href="javascript:void(0);">收藏该文</a>    <a id="green_channel_weibo" href="javascript:void(0);" title="分享至新浪微博" onclick="ShareToTsina()"><img src="//common.cnblogs.com/images/icon_weibo_24.png" alt=""></a>    <a id="green_channel_wechat" href="javascript:void(0);" title="分享至微信" onclick="shareOnWechat()"><img src="//common.cnblogs.com/images/wechat.png" alt=""></a></div><div id="author_profile">    <div id="author_profile_info" class="author_profile_info">            <a href="http://home.cnblogs.com/u/Wayou/" target="_blank"><img src="//pic.cnblogs.com/face/431064/20140914104240.png" class="author_avatar" alt=""></a>        <div id="author_profile_detail" class="author_profile_info">            <a href="http://home.cnblogs.com/u/Wayou/">刘哇勇</a><br>            <a href="http://home.cnblogs.com/u/Wayou/followees">关注 - 10</a><br>            <a href="http://home.cnblogs.com/u/Wayou/followers">粉丝 - 2494</a>        </div>    </div>    <div class="clear"></div>    <div id="author_profile_honor">荣誉:<a href="http://www.cnblogs.com/expert/" target="_blank">推荐博客</a></div>    <div id="author_profile_follow">                <a href="javascript:void(0);" onclick="follow('c5e8573b-1ed6-e111-aa3f-842b2b196315');return false;">+加关注</a>    </div></div><div id="div_digg">    <div class="diggit" onclick="votePost(3663567,'Digg')">        <span class="diggnum" id="digg_count">65</span>    </div>    <div class="buryit" onclick="votePost(3663567,'Bury')">        <span class="burynum" id="bury_count">0</span>    </div>    <div class="clear"></div>    <div class="diggword" id="digg_tips">    </div></div></div><div class="clear"></div><div id="post_next_prev"><a href="http://www.cnblogs.com/Wayou/p/things_you_dont_know_about_frontend.html" class="p_n_p_prefix">« </a> 上一篇:<a href="http://www.cnblogs.com/Wayou/p/things_you_dont_know_about_frontend.html" title="发布于2014-04-10 19:58">前端不为人知的一面--前端冷知识集锦</a><br><a href="http://www.cnblogs.com/Wayou/p/highlight_code_with_sublimetext_style.html" class="p_n_p_prefix">» </a> 下一篇:<a href="http://www.cnblogs.com/Wayou/p/highlight_code_with_sublimetext_style.html" title="发布于2014-04-14 17:24">如何在博客中使用SublimeText风格的代码高亮样式</a><br></div></div>

</div><div class="postDesc">posted @ <span id="post-date">2014-04-14 11:17</span> <a href="http://www.cnblogs.com/Wayou/">刘哇勇</a> 阅读(<span id="post_view_count">184768</span>) 评论(<span id="post_comment_count">36</span>)  <a href="https://i.cnblogs.com/EditPosts.aspx?postid=3663567" rel="nofollow">编辑</a> <a href="#" onclick="AddToWz(3663567);return false;">收藏</a></div></div><script type="text/javascript">var allowComments=true,cb_blogId=123544,cb_entryId=3663567,cb_blogApp=currentBlogApp,cb_blogUserGuid='c5e8573b-1ed6-e111-aa3f-842b2b196315',cb_entryCreatedDate='2014/4/14 11:17:00';loadViewCount(cb_entryId);var cb_postType=1;</script></div>

原文地址:https://www.cnblogs.com/Vayne-N/p/8094680.html