我们都知道,在javascript中通过var obj = {name: "dancy"}
可直接创建一个对象。
当我们需要取值时,直接使用obj.name
就能获取到该对象name属性的值。
当我们需要修改对象的属性值也只要 obj.name = "lotus"
就能实现我们想要的结果。
当我们需要删除该对象的属性时,直接 delete obj.name
就可以了。
当我们需要获取对象所有的属性和值,直接使用 for in
即可。
我们对于对象的各种操作已经非常熟悉了,但,在对对象进行各种操作的同时,是否有想过:
在ECMAScript 中有两种属性:数据属性和访问器属性。当我们弄懂什么是数据属性,什么是访问器属性的时候,上面的那些问题就不复存在了。
数据属性有四个,它们分别是 configurable、enumerable、writable、value
。对于这几个数据属性的定义,这里引用《javascript高级程序设计(第3版)》中的描述。
configurable: 表示能否通过 delete 删除属性从而重新定义属性,能否修改属性的特性,或者能否把属性修改为访问器属性。默认为true。
enumerable:表示能否通过 for-in 循环返回属性。默认为true。
writable: 表示能否修改属性的值。默认为true。
value:属性的数据值。默认为undefined。
接下来,我们尝试理解上面这些描述的含义。首先,我们先创建一个如下所示的对象:
var obj = {
name: "dancy",
age: 23
}
根据configurable的定义,我们通过 Object.defineProperty
修改obj的name属性的数据属性configurable
为false。
Object.defineProperty(obj, "name", {
configurable: false
});
根据定义, configurable为false时,我们不能通过 delete 删除属性,不能修改属性的特性,不能把属性修改为访问器属性。接下来,我们做以下尝试:
尝试二:修改属性的特性
在控制台测试结果:
当我们把对象属性的configurable属性改为false之后,就不能再将其改回true了,同时对于enumerable属性的修改也受到限制,当我们尝试修改时,会报错!
根据定义,我们重新定义obj对象并修改其name属性的enumerable为false,然后用for in遍历obj对象。
Object.defineProperty(obj, "name", {
enumerable: false
});
for(var key in obj) {
console.log(key + "——" + obj[key])
}
控制台测试结果:
因为属性name的enumerable改为了false,所以在使用for in
遍历obj对象时,我们无法遍历到name属性,所以上面的代码只打印出了属性age。
value属性的值能否被修改是根据writable确定的,如writable为true,则value可以被修改,否则不能被修改。当writable为false时尝试修改value同样会报错。
我们将writable改为false,然后修改name的值为lotus,再控制台输出结果如下:
所以,当writable为false时,我们无法修改name属性的值。这里需要注意的是,如果name属性的configurable
为false,此时我们可以将writable的值改为false,但如果再想将writable改为true时会报错!!!
但如果configurable为true的话,我们就可随意修改writable。
访问器属性也有四个,它们分别是 configurable、enumerable、get、set
。其中configurable和enumerable的作用和数据属性中的作用相同,这里不再继续说明。
get:在读取属性时调用的函数。默认值为 undefined。
set:在写入属性时调用的函数。默认值为 undefined。
同理,我们可以使用 Object.defineProperty
对访问器属性进行定义。以上面的obj对象为例。
Object.defineProperty(obj, "name", {
get: function(){
return this._name;
},
set: function(newValue){
if(typeof newValue === 'number') {
this._name = 'this is a number';
} else {
this._name = newValue;
}
}
});
细心的小伙伴肯定发现了,我在定义get和set使用的是_name
,而不是直接使用的name,这是因为我们在obj.name = 123
触发set函数时,set函数内部同时也触发了get方法,所以导致栈溢出错误,无法直接使用name,具体可以参考这篇文章 https://www.jianshu.com/p/043c11ec2e33 。
需要注意的是,访问器属性不能直接定义,必须使用 Object.defineProperty()来定义。
还有就是,我们不一定非要同时定义get和 set。但只指定 get 意味着属性是不能写,尝试写入属性会被忽略。在严格模式下,尝试写入只指定了 get函数的属性会抛出错误。
类似地,只指定 set函数的属性也不能读,否则在非严格模式下会返回 undefined,而在严格模式下会抛出错误。
当我们需要对属性取值或修改行为进行控制时,控制访问器属性set和get变能满足我们的需求了。