对象属性有两种

  • 数据属性
  • 访问器属性

数据属性:有四个特性

  • [[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高级程序设计一书。