js对象可枚举属性以及forin循环和forof循环
js中每个对象的属性(js⾥万物皆属性,对象的属性也是对象)都有⼀个属性叫enumerable(可枚举性),这个属性true/false决定了该对象的属性是否可枚举(就是让⼀些⽅法访问到这个属性)。
js中哪些属性可枚举,哪些不可枚举?
1、可枚举属性:可以⽤⽅法来判断,返回Boolean,但是通过原型链继承的属性除外。
2、js基本数据类型⾃带的原型属性不可枚举。
3、通过Object.defineProperty()⽅法指定enumeralbe为false的属性不可枚举。
针对第⼀点看这个:我们把可⽤propertyIsEnumberable()检测“lastName”,返回 true,检测“firstName”属性是通过原型链继承得到,所以检测的时候是 false,,说明检测继承得到的属性的时候会失效,所以⽤这个⽅法的时候要注意不可⽤于继承得到的属性是否是可枚举属性,for in 遍历可枚举属性的就不会发⽣这样的情况,⼤可⽅法去⽤for in 检测⼀个对象的可枚举属性。
针对第⼆点,看个例⼦:
var obj = new Object();
for(var key in obj) {
console.log("obj." + key + " = " + obj[key]);
}
// 输出为空
Object, Array, Number等,这种js内置的原型属性不可枚举。
再看第三个例⼦,定义了某个属性的enumberable属性为false,这样会让for in 也遍历不到,外界就⽆法对这个属性读写了:
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
可见,这两种⽅法,Object.keys()只能返回对象本⾝具有的可枚举属性
JSON.stringify()只能返回对象本⾝的可枚举属性,并序列化为JSON字符串对象。
再来看看for in 和for of 循环的异同:
所以这⾥,你得弄明⽩什么是可枚举属性,什么是Iterator接⼝主要供f消费。
考别⼈的代码,说明⾃⼰的问题:
Object.prototype.objCustom = function() {};
Array.prototype.arrCustom = function() {};
let iterable = [3, 5, 7];
iterable.foo = 'hello';
每个对象将继承objCustom属性,并且作为Array的每个对象将继承arrCustom属性,因为将这些属性添加到Object.prototype和Array.prototype。由于继承和原型链,对象iterable继承属性objCustom和arrCustom。
for (let i in iterable) {
console.log(i); // 0, 1, 2, "foo", "arrCustom", "objCustom"
}
此循环仅以原始插⼊顺序记录iterable对象的可枚举属性。它不记录数组元素3, 5, 7或hello,因为这些不是枚举属性。但是它记录了数组索引以及arrCustom和objCustom。
for (let i in iterable) {
if (iterable.hasOwnProperty(i)) {
console.log(i); // 0, 1, 2, "foo"
}
js arguments}
这个循环类似于第⼀个,但是它使⽤hasOwnProperty() 来检查,如果到的枚举属性是对象⾃⼰的(不是继承的)。如果是,该属性被记录。记录的属性是0, 1, 2和foo,因为它们是⾃⾝的属性(不是继承的)。属性arrCustom和objCustom不会被记录,因为它们是继承的。
for (let i of iterable) {
console.log(i); // 3, 5, 7
}
该循环迭代并记录iterable作为可迭代对象定义的迭代值,只检测具有数组键名的属性值,这⾥就是数
组元素3, 5, 7,⽽不是任何不具有数据键名的属性。
for (var n of [1,2,3,4,5,6]) {
if (n > 3)
break;
console.log(n);
}
结果:
当n > 3的时候,终⽌of 循环。
所以for of使⽤起来⽐for in要更好⼀些。⼤家在⼯作中,多多使⽤,掌握好它。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论