理解JavaScript中的可枚举与不可枚举属性

1.在JavaScript中,对象的属性分为可枚举和不可枚举之分,它们是由属性的enumerable值决定的。可枚举性决定了这个属性能否被for…in查找遍历到。
2.for…in循环是 遍历对象的每一个可枚举属性,包括原型链上面的可枚举属性,而Object.keys()只是遍历自身的可枚举属性,不可以遍历原型链上的可枚举属性. 这是for…in和Object.keys()的主要区别,而Object.getOwnPropertyNames()则是遍历自身所有属性(不论是否是可枚举的),不包括原型链上面的.
一、怎么判断属性是否可枚举

js中基本包装类型的原型属性是不可枚举的,如Object, Array, Number等,如果你写出这样的代码遍历其中的属性:

var num = new Number();
for(var pro in num) {
    console.log("num." + pro + " = " + num[pro]);
}

它的输出结果会是空。这是因为Number中内置的属性是不可枚举的,所以不能被for…in访问到。

Object对象的propertyIsEnumerable()方法可以判断此对象是否包含某个属性,并且这个属性是否可枚举。

需要注意的是:如果判断的属性存在于Object对象的原型内,不管它是否可枚举都会返回false。

二、枚举性的作用

属性的枚举性会影响以下三个函数的结果:

for…in

Object.keys()

JSON.stringify

先看一个例子,按如下方法创建kxy对象:

function Person() {
    this.name = "KXY";
}
Person.prototype = {
    constructor: Person,
    job: "student",
};
 
var kxy = new Person();
Object.defineProperty(kxy, "sex", {
    value: "female",
    enumerable: false
});

其中用defineProperty为对象定义了一个名为”sex”的不可枚举属性

接下来做以下验证:

for(var pro in kxy) {
    console.log("kxy." + pro + " = " + kxy[pro]);
  }

遍历结果:

kxy.name = KXY

kxy.constructor = function Person() {
this.name = "KXY";
}
kxy.job = student

可以看到除了”sex“之外的属性都遍历到了

[ 个人补充

  for(var pro in kxy) {
    if(kxy.hasOwnProperty(pro)) console.log("kxy." + pro + " = " + kxy[pro]);
  }
  打印结果是:

  kxy.name = KXY

]
console.log(Object.keys(kxy));
结果:
["name"]

只包含”name”属性,说明该方法只能返回对象本身具有的可枚举属性。

console.log(JSON.stringify(kxy));
结果:
{"name":"KXY"}

此方法也只能读取对象本身的可枚举属性,并序列化为JSON对象。

function People(){

  this.name = 'Tom';

  this.age = 24;

  this.c = function(){

  console.log('1');
  }
}
People.prototype.look = function(){
  console.log('look at this');
}
var b = new People();
//使用Object.defineProperty方法为b添加一个'sex'的不可枚举属性
Object.defineProperty(b,'sex',{
value:'女',
enumerable:false
});
for(var i in b){
  console.log(i);
}

我们可以遍历到对象b的所有可枚举属性,其中包括从原型继承来的look.

接着我们使用Object.keys()来遍历,代码如下:

function People(){

  this.name = 'Tom';

  this.age = 24;

  this.c = function(){

  console.log('1');
  }
}
People.prototype.look = function(){
  console.log('look at this');
}
var b = new People();
//使用Object.defineProperty方法为b添加一个'sex'的不可枚举属性
Object.defineProperty(b,'sex',{
value:'女',
enumerable:false
});
console.log(Object.keys(b));//结果["name","age","c"]

从打印的结果我们看到, Object.keys()遍历的是自身可枚举属性,他不包括原型链上的look.

下面我们使用Object.getOwnPropertyNames()方法试试,具体代码如下:

function People(){

  this.name = 'Tom';

  this.age = 24;

  this.c = function(){

  console.log('1');
  }
}
People.prototype.look = function(){
  console.log('look at this');
}
var b = new People();
//使用Object.defineProperty方法为b添加一个'sex'的不可枚举属性
Object.defineProperty(b,'sex',{
value:'女',
enumerable:false
});
console.log(Object.getOwnPropertyNames(b));//结果["name", "age", "c", "sex"]

可以看到,Object.getOwnPropertyNames()遍历的是自身的可枚举和不可枚举属性, 不含原型链上的.

  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值