对象属性有两种
- 数据属性
- 访问器属性
数据属性:有四个特性
- [[configurable]](代表属性是否可配置):当这个属性描述符值为true时,该属性可能会被改变,也可能会被从相应的对象删除,默认为false。
- [[enumerable]](代表属性是否可枚举):表示能否通过for - in循环返回属性,默认为false。
- [[value]](代表属性值):包含这个属性的数据值,读取属性值的时候,从这个位置读;写入属性时,把新的值保存在这个位置,默认为undefined.
- [[writable]](代表属性是否可写):表示能否修改属性的值。
修改属性的特性必须使用Object.defineProperty()。包含三个参数:属性所在的对象,属性的名字,和一个描述符对象。
1 var obj = {
2 x:123,
3 y:456
4 }
5 Object.defineProperty(obj,"x",{
6 value:56,
7 writable:true, //表示可以重新赋值写入
8 enumerable:false, //表示不能被枚举
9 configurable:false //表示不能被配置
10 });
11 console.log(obj.x); //56
12 obj.x = 10;
13 console.log(obj.x); //不能被重新赋值,输出56
14 for(var i in obj){
15 console.log(i); //输出 y
16 }
configurable:false后:
- delete obj.x 或对象其他属性无法删除
- 这个属性后面无法再使用Object.defineProperty()方法重新配置configurable:true。
- 有一个特殊情况:writable:true可以改为false。但是writable:false无法改为true。
1 var obj = {
2 x:123
3 }
4 //等价于
5 Object.defineProperty(obj,"x",{
6 value:123,
7 writable:true,
8 enumerable:true,
9 configurable:true
10 });
1 var obj = {
2 x:123
3 }
4 Object.defineProperty(obj,"y",{value:123});
用Object.defineProperty()添加一个新的值时,其他三个属性不写,默认为false,如果只是调用此方法修改已定义的属性特性则无此限制。上面代码等价于:
1 var obj = {
2 x:123
3 }
4 Object.defineProperty(obj,"y",{
5 value:123,
6 writable:false,
7 enumerable:false,
8 configurable:false
9 });
可以使用Object.defineProperties()同时设置多个属性
1 var obj = {};
2 Object.defineProperties(obj,{
3 x:{
4 value: "yewenxiang",
5 writable:true
6 },
7 y:{
8 value: "xiangwang",
9 writable:false
10 }
11 });
访问器属性:有四个特性
- [[configurable]]:表示能否通过delete删除属性从而重新定义属性,能否修改属性的特性,能否把属性修改为数据属性。对于直接在对象定义的属性,这个特性的默认值为true。
- [[enumerable]]:表示能否通过for - in 循环返回属性。对于直接在对象定义的属性,这个特性的默认值为true。
- [[get]]:在读取属性时调用的函数。默认值为undefined
- [[set]]:在写入属性时调用的函数。默认值为undefined
使用访问器属性的常见方式:设置一个属性的值导致其他属性发生变化
1 var obj = {
2 x:2016,
3 y:12
4 };
5 Object.defineProperty(obj,"z",{
6 get:function(){
7 return this.x;
8 },
9 set:function(newValue){
10 if(newValue > 12){
11 this.x = 2017;
12 this.y = 1;
13 }
14 }
15 })
16 console.log(obj.x); //2016
17 console.log(obj.y); //12
18 obj.z = 13;
19 console.log(obj.x); //2017
20 console.log(obj.y); //1
21 console.log(obj.z); //2017
不一定非要同时指定get和set。只设置get代表属性z是不可写的,只能获取,尝试写入会被忽略,严格模式下或报错。同样只设置set代表属性z是不可读的,只能设置,尝试读取会返回undefined,在严格模式会报错。
访问器属性另外一种写法:
1 var obj = {
2 x:1,
3 y:2,
4 z:3,
5 get zhouchang(){
6 return this.x+this.y+this.z;
7 },
8 set change(value){
9 this.x*=value;
10 this.y*=value;
11 this.z*=value;
12 }
13 };
14 console.log(obj.zhouchang); //6
15 obj.change = 2;
16 console.log(obj.zhouchang); //12
最后可以使用Object.getOwnPropertyDescriptor()方法来获取属性特性的值:有两个参数,属性所在的对象和要被读取其描述符的属性名
- 访问器属性eg:
Object.getOwnPropertyDescriptor(obj,"x").configurable或.enumerable或.get或.set
- 数据属性eg:
Object.getOwnPropertyDescriptor(obj,"z").configurable或.enumerable或.writable或.value
----总结来自javascript高级程序设计一书。