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]);
}