ECMAScript中有两种属性:数据属性和访问器属性
1. 数据属性
数据属性指包含一个数据值的位置,可在该位置读取或写入值,该属性有4个供述其行为的特性:
1. [[configurable]]:表示能否使用delete操作符删除从而重新定义,或能否修改为访问器属性。默认为true;
2. [[Enumberable]]:表示是否可通过for-in循环返回属性。默认true;
3. [[Writable]]:表示是否可修改属性的值。默认true;
4. [[Value]]:包含该属性的数据值。读取/写入都是该值。默认为undefined;如上面实例对象Person中定义了name属性,其值为’wenzi’,对该值的修改都反正在这个位置
要修改对象属性的默认特征(默认都为true),可调用Object.defineProperty()方法,它接收三个参数:属性所在对象,属性名和一个描述符对象,必须是:configurable、enumberable、writable和value,可设置一个或多个值
var person = {};
Object.defineProperty(
person, 'name', {
configurable: false, // 不可删除,且不能修改为访问器属性
writable: false, // 不可修改
value: 'wenzi' // name的值为wenzi
}
);
console.log(person.name); // wenzi
console.log(delete person.name); // false,无法删除
person.name = 'lily';
console.log(person.name); // wenzi
2. 访问器属性
访问器属性主要包括一对getter和setter函数,在读取访问器属性时,会调用getter返回有效值;写入访问器属性时,调用setter,写入新值;该属性有以下4个特征:
1. [[Configurable]]:是否可通过delete操作符删除重新定义属性;
2. [[Numberable]]:是否可通过for-in循环查找该属性;
3. [[Get]]:读取属性时自动调用,默认:undefined;
4. [[Set]]:写入属性时自动调用,默认:undefined;
要修改对象属性的默认特征(默认都为true),可调用Object.defineProperty()方法,它接收三个参数:属性所在对象,属性名和一个描述符对象,必须是:configurable、enumberable、writable和value,可设置一个或多个值
访问器属性不能直接定义,必须使用defineProperty()来定义,如下:
var person = {
_age: 18
};
Object.defineProperty(person, 'isAdult', {
Configurable : false,
get: function () {
if (this._age >= 18) {
return true;
} else {
return false;
}
}
});
console.log( person.isAdult ); // true
Object.defineProperty()方法设置属性时,不能同时声明访问器属性(set和get)和数据属性(writable或者value)。意思就是,某个属性设置了writable或者value属性,那么这个属性就不能声明get和set了,反之亦然。
如若像下面的方式进行定义,访问器属性和数据属性同时存在:
var o = {};
Object.defineProperty(o, 'name', {
value: 'wenzi',
set: function(name) {
myName = name;
},
get: function() {
return myName;
}
});
上面的代码看起来貌似是没有什么问题,但是真正执行时会报错,报错如下:
Uncaught TypeError: Invalid property. A property cannot both have accessors and be writable or have a value
对于数据属性,可以取得:configurable,enumberable,writable和value;
对于访问器属性,可以取得:configurable,enumberable,get和set。
3. 获取Configurable值
ES5为我们提供了Object.getOwnPropertyDescriptor(object, property)来获取内部属性
var person = {name:'wenzi'};
var desp = Object.getOwnPropertyDescriptor(person, 'name'); // person中的name属性
console.log( desp );// {value: "wenzi", writable: true, enumerable: true, configurable: true}
console.log( person.name ); // wenzi
console.log( delete person.name ); // true,删除成功
console.log( person.name ); // undefined
通过Object.getOwnPropertyDescriptor(object, property)我们能够获取到4个内部属性,configurable控制着变量或属性是否可被删除。这个例子中,person.name的configurable是true,则说明是可以被删除的.
4. 面试题
a = 1;
var desp = Object.getOwnPropertyDescriptor(window, 'a');
console.log( desp.configurable ); // true,可以删除
var b = 2;
var desp = Object.getOwnPropertyDescriptor(window, 'b');
console.log( desp.configurable ); // false,不能删除