我在 Mozilla 开发网络 (MDN) 上进行了一些研究。
原来对象有一个obj.propertyIsEnumerable(prop) 方法来检查属性是否可枚举。从 MDN 中给出的示例来看,通过原型链继承的属性是不可枚举的,因此该方法对这些属性返回 false。
不可枚举属性的一个例子是构造函数和数组的属性length。
对于问题的可迭代部分,我将引用 MDN:“ECMAScript 6 中的可迭代是一个接口(或者换句话说,一个协议),而不是像 Array 或 Map 这样的对象类型”。
也就是说,对象需要实现这个接口,才能使其属性可迭代。对于 String、Array、Map、Set、WeakSet 和 Generator 对象等内置可迭代对象就是这种情况。
以下代码说明了这一点:
var aString = "hello"
typeof aString[Symbol.iterator] // "function"
aString[Symbol.iterator]() + "" // "[object String Iterator]"
[...aString] // ["h", "e", "l", "l", "o"]
当然,您可以定义自己的迭代器实现。
回到问题,附加到对象或其原型的属性(直接,意味着不通过继承)将在for...in 循环中显示为可枚举。对于可迭代对象,您需要一个实现,可以是前面提到的实现,也可以是您自己的实现。这是来自 MDN 的一个很好的例子。
let arr = [ 3, 5, 7 ];
arr.foo = "hello";
for (let i in arr) {
console.log(i); // logs "0", "1", "2", "foo" these are property names or indexes
}
for (let i of arr) {
console.log(i); // logs "3", "5", "7" these are actual values of an
// iterable implementation provided by the array prototype
}
let 关键字在此上下文中等同于 var 定义(尽管它具有更多含义,但它们超出了本文的范围)。
如您所见,数组已经实现了iterable 接口(来自数组原型),因此在使用for...of 循环时它会产生它的值,而foo 属性没有显示(两者都没有)值或属性名称)。
因此,如果您的对象没有实现 iterable 接口,则它不应该是可迭代的(原则上),因此它将在 for...of 循环中显示其属性。