js中的可枚举属性

思考这样一个问题,假如你定义了一个Array

var a = [1,2,3,4,5];

然后你对这个数组进行了for...in循环求和操作

var total = null;
for (var i in a) {
   total += a[i];
}
cosole.log(total); //15

过了一段时间,你的teammate为了解决一个排序问题,像Array函数添加了一个原型方法order

Array.prototype.order = function() {
    //排序操作
};

这时候你再去执行你的求和操作,会发现结果不再是15了而是



原因是Array数组的原型方法也被遍历出来了,这里就涉及到js可枚举属性了。

js中什么是可枚举属性######

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

js中哪些属性可枚举,哪些不可枚举?######
  1. js基本数据类型自带的原型属性不可枚举。
  2. 通过Object.defineProperty()方法指定enumeralbe为false的属性不可枚举。
js中哪些方法会访问到enumerable为true的对象属性?######
  1. for...in操作
  2. Object.keys()方法(和for...in的区别就是Object.keys()不会返回对象原型链上的属性)
  3. JSON.stringify()方法。
    虽然enumerable为false的属性上述三个方法不能访问到,但是你要是直接获取的话,还是能直接获取到的。

回过头来看最开始的情形,你总不能要求别人把它定义的原型方法删除,这时候就要用到object.hasOwnProperty()方法了。
该方法会判断一个对象是否含有指定的属性,但是会忽略从原型链上继承的属性。

将开头的代码改成如下:

var total = null;
for (var i in a) {
    if (a.hasOwnProperty(i)) {
        total += a[i];
    }
}
cosole.log(total); //15

现在就没有原先那样奇怪的结果了。

你可能感兴趣的:(js中的可枚举属性)