ES6迭代器和for-of循环

如何循环一个数组?20年前 JavaScript 诞生的时候,你会这么写:

for (var index = 0; index < myArray.length; index++) {  
  console.log(myArray[index]);
}

ES5 之后,可以使用内置的 forEach 方法:

myArray.forEach(function (value) {  
  console.log(value);
});

这样就稍微短了点了,但是仍然有一个小的缺点:无法使用 break 语句跳出循环,或者使用 return 从函数体内返回。

要是有可以用 for 循环的语法遍历数组的所有元素,那该多好。

那么for-in 循环如何?

for (var index in myArray) { // 不要真的这样写  
  console.log(myArray[index]);
}

这样写有以下几个问题:

代码中赋值为index的值是字符串"0","1","2"等,而不是真是的数字。由于你不想要碰到字符串计算("2" + 1 == "21")的状况,这对于编程而言是极其不方便的。

循环体不仅仅会遍历数组元素,还会遍历任意其他的自定义添加的属性。例如,如果数组包含了一个不能枚举的属性 myArray.name,那么这次循环就会在 index == "name" 的时候额外执行一遍。甚至数组原型链上的属性也都会被遍历到。

最让人感到惊奇的是,在某些状况下,这段代码会以随机顺序循环数组元素。

简而言之,for-in 循环在设计之初就是用于普通的以字符串为 key 值的对象的语法,而不适用与数组。

强大的 for-of 循环 让我们来看看 for-of 循环:

for (var value of myArray) {  
  console.log(value);
}

唔,上述代码就是看起来并没有很强大,对吗?好吧,我们之后会探索 for-of 循环隐藏的强大之处。就现在而言,只需要记住:

这是最简洁、直白的循环数组元素的方法 可以避免所有 for-in 循环的陷阱 不同于 forEach(),可以使用 break, continue 和 return for-in 循环用以遍历对象的属性。

for-of 循环用以遍历数据 -- 就像数组中的值一样

但这还不是所有的内容。

其他集合也支持 for-of 进行遍历 for-of 循环不仅仅支持数组的遍历。同样适用于很多类似数组的对象,例如 DOM NodeList。

它也支持字符串的遍历,会把字符串作为一组 Unicode 的字符进行遍历:

for (var chr of "😺😲") {  
  alert(chr);
}

它也可以应用于 Map 和 Set 对象(ES6中新增的数据结构,之后的文章中会提及)。

例如,Set 对象可以有效的去重:

// 从一个单词组成的数组声明一个新的 Set
var uniqueWords = new Set(words);  

然后你就轻松可以遍历 Set 中的内容了:

for (var word of uniqueWords) {  
  console.log(word);
}

Map 则稍微有点不同:Map 中的数据是由键值对组成的,所以你需要使用将其中的键值解构为两个独立的变量:

for (var [key, value] of phoneBookMap) {  
  console.log(key + "'s phone number is: " + value);
}

解构 (Destructing) 也是 ES6 中的特性,同样会在之后的文章中提及。

到现在为止,你可能已经可以想象到: JavaScript 已经有了一些新的集合类型,且在不久的将来会出现更多。而 for-of 则是被设计出来用以在这些集合上使用的循环语句。

for-of 并不适用于处理原有的原生对象,但是如果你想要遍历对象的属性,可以使用 for-in 或者内置的 Object.keys():

// 输出对象自身可以枚举的值
for (var key of Object.keys(someObject)) {  
  console.log(key + ": " + someObject[key]);
}