# 迭代器和生成器
# 可迭代的
如果一个对象具有 Symbol.iterator
属性的实现,则该对象被认为是可迭代的。一些内置类型,如 Array
、Map
、Set
、String
、Int32Array
、Uint32Array
等,它们的 Symbol.iterator
属性已经实现。对象上的 Symbol.iterator
函数负责返回要迭代的值列表。
# Iterable 接口
如果我们想采用上面列出的可迭代类型,Iterable
是我们可以使用的类型。这是一个例子:
function toArray<X>(xs: Iterable<X>): X[] {
return [...xs]
}
# for..of 语句
for..of
循环遍历一个可迭代对象,调用该对象的 Symbol.iterator
属性。这是一个数组上的简单 for..of
循环:
let someArray = [1, "string", false];
for (let entry of someArray) {
console.log(entry); // 1, "string", false
}
# for..of 与 for..in 语句
for..of
和 for..in
语句都遍历列表;但是,迭代的值不同,for..in
返回被迭代对象的键列表,而 for..of
返回被迭代对象的数字属性的值列表。
这是一个演示这种区别的示例:
let list = [4, 5, 6];
for (let i in list) {
console.log(i); // "0", "1", "2",
}
for (let i of list) {
console.log(i); // 4, 5, 6
}
另一个区别是 for..in
对任何对象进行操作;它用作检查此对象的属性的一种方式。另一方面,for..of
主要对可迭代对象的值感兴趣。像 Map
和 Set
这样的内置对象实现了 Symbol.iterator
属性,允许访问存储的值。
let pets = new Set(["Cat", "Dog", "Hamster"]);
pets["species"] = "mammals";
for (let pet in pets) {
console.log(pet); // "species"
}
for (let pet of pets) {
console.log(pet); // "Cat", "Dog", "Hamster"
}
# 代码生成
# 针对 ES5 和 ES3
当以 ES5 或 ES3 兼容引擎为目标时,迭代器仅允许用于 Array
类型的值。对非数组值使用 for..of
循环是错误的,即使这些非数组值实现了 Symbol.iterator
属性。
编译器将为 for..of
循环生成一个简单的 for
循环,例如:
let numbers = [1, 2, 3];
for (let num of numbers) {
console.log(num);
}
将生成为:
var numbers = [1, 2, 3];
for (var _i = 0; _i < numbers.length; _i++) {
var num = numbers[_i];
console.log(num);
}
# 以 ECMAScript 2015 及更高版本为目标
当针对 ECMAScipt 2015 兼容引擎时,编译器将生成 for..of
循环以针对引擎中的内置迭代器实现。