js 对象可枚举属性以及for in 循环和for of 循环

js中每个对象的属性(js里万物皆属性,对象的属性也是对象)都有一个属性叫enumerable(可枚举性),这个属性true/false决定了该对象的属性是否可枚举(就是让一些方法访问到这个属性)。

js中哪些属性可枚举,哪些不可枚举?

1、js基本数据类型自带的原型属性不可枚举。

2、通过Object.defineProperty()方法指定enumeralbe为false的属性不可枚举。

看个例子:

var obj = new Object();
for(var key in obj) {
    console.log("obj." + key + " = " + obj[key]);
}
// 输出为空

Object, Array, Number等,这种js内置的原型属性不可枚举。

再看第二个例子:

function Person() {
    this.name = "zhangsan";
}
Person.prototype = {
    constructor: Person,
    job: "student",
};
 
var person = new Person();
Object.defineProperty(person , "sex", {
    value: "man",
    enumerable: false // 定义了一个不可枚举的属性
});
for(var key in person) {
    console.log("person." + key + " = " + person[key]);
}

结果我们只看到了name、constructor、job属性,我们定义的“sex”属性设置了enumberable: false,是不可枚举属性,所以for in 遍历不到。

 不可枚举属性影响了for in 的结果,它还能影响什么操作结果呢?

Object.keys()和JSON.stringify

扫描二维码关注公众号,回复: 5800127 查看本文章

可见,这两种方法,Object.keys()只能返回对象本身具有的可枚举属性
JSON.stringify()只能返回对象本身的可枚举属性,并序列化为JSON字符串对象。

 再来看看for in 和for of 循环的异同:

for...in,es5标准,循环会遍历一个object所有的可枚举属性,遍历(当前对象及其原型上的)每一个属性名称key

for...of,es6标准,会遍历具有iterator接口(类似数组,比如ArrayMapSetStringTypedArray,arguments 对象等等)的数据结构,遍历(当前对象上的)每一个属性值value。

所以这里,你得弄明白什么是可枚举属性,什么是iterator接口???Iterator接口主要供for...of消费。

考别人的代码,说明自己的问题:

Object.prototype.objCustom = function() {};
Array.prototype.arrCustom = function() {}; 
 
let iterable = [3, 5, 7]; 
iterable.foo = 'hello';

每个对象将继承objCustom属性,并且作为Array的每个对象将继承arrCustom属性,因为将这些属性添加到Object.prototypeArray.prototype。由于继承和原型链,对象iterable继承属性objCustomarrCustom

for (let i in iterable) {
  console.log(i); // 0, 1, 2, "foo", "arrCustom", "objCustom" 
}

此循环仅以原始插入顺序记录iterable 对象的可枚举属性。它不记录数组元素357 或hello,因为这些不是枚举属性。但是它记录了数组索引以及arrCustomobjCustom

for (let i in iterable) {
  if (iterable.hasOwnProperty(i)) {
    console.log(i); // 0, 1, 2, "foo"
  }
}

这个循环类似于第一个,但是它使用hasOwnProperty() 来检查,如果找到的枚举属性是对象自己的(不是继承的)。如果是,该属性被记录。记录的属性是0, 1, 2和foo,因为它们是自身的属性(不是继承的)。属性arrCustom和objCustom不会被记录,因为它们是继承的。

for (let i of iterable) {
  console.log(i); // 3, 5, 7 
}

该循环迭代并记录iterable作为可迭代对象定义的迭代值,这些是数组元素 357,而不是任何对象的属性。

猜你喜欢

转载自www.cnblogs.com/whq920729/p/10663448.html
今日推荐