ECMAScript 5中的属性描述符详解_基础知识

属性描述符是ES5中新增的概念,其作用是给对象的属性增加更多的控制。

Object.defineProperty

要研究属性描述符,首先要谈谈 Object.defineProperty 方法。这个方法的作用是给对象定义新属性或修改已存在的属性。其原型如下:

复制代码 代码如下:

Object.defineProperty(obj, prop, descriptor)

使用示例:

复制代码 代码如下:

var obj = { };
Object.defineProperty(obj, 'attr', { value: 1 });

上面一段代码给obj对象增加了一个名为attr的属性,值为1。相当于:

复制代码 代码如下:

var obj = { };
obj.attr = 1;

相比起来,Object.defineProperty 的写法看似更为复杂。但是,它最大的奥秘在于其第三个参数。

数据描述符

假设我们希望attr是一个只读属性,就可以加上 writable 数据描述符:

复制代码 代码如下:

var obj = { };
Object.defineProperty(obj, 'attr', {
    value: 1,
    writable: false
});
console.log(obj.attr);
obj.attr = 2; // fail
console.log(obj.attr);

执行以上程序可以发现,两次打印出来的attr的值都是1,也就是说对属性的写入失败。然而,这样的结果会有点莫名其妙,因为赋值语句的执行没有异常,却失败了,试想如果在大片的代码中出现这样的问题,就很难排查出来。事实上,只要以严格模式运行代码,就会产生异常:

复制代码 代码如下:

'use strict'; // 进入严格模式
var obj = { };
Object.defineProperty(obj, 'attr', {
    value: 1,
    writable: false
});
obj.attr = 2;  // throw exception

下面再来看看另一个数据描述符 enumerable ,它可以控制属性是否能被枚举。如果只是简单地定义一个属性,这个属性是可以在for...in循环中被枚举出来的:

复制代码 代码如下:

var obj = { };
obj.attr = 1;
for (var i in obj) { console.log(obj[i]); }
enumerable 可以将其“藏”起来:

var obj = { };
Object.defineProperty(obj, 'attr', {
    value: 1,
    enumerable: false
});
for (var i in obj) { console.log(obj[i]); }

执行上面一段代码,会发现控制台什么也没输出,因为此时attr属性无法被枚举了。

讲到这里,大家可能有一个疑问,属性描述符能否被修改?比方说一个只读属性是否可以再次定义为可写?其实这取决于另一个数据描述符 configurable ,它可以控制属性描述符能否被更改。

复制代码 代码如下:

var obj = { };
Object.defineProperty(obj, 'attr', {
    value: 1,
    writable: false,
    configurable: true
});
Object.defineProperty(obj, 'attr', {
    writable: true
});
obj.attr = 2;

上面一段代码先把attr定义为只读属性,然后又重新定义为可写。所以对attr的写入是成功的。

存取描述符

存取描述符类似面向对象中的get/set访问器。

复制代码 代码如下:

var obj = { };
Object.defineProperty(obj, 'attr', {
    set: function(val) { this._attr = Math.max(0, val); },
    get: function() { return this._attr; }
});
obj.attr = -1;
console.log(obj.attr); // 0

在上面一段代码中,对attr的访问事实上变成了对_attr的访问,而且在set函数中限制了最小值为0。

获取属性描述符

前面所述都是设置属性描述符,那如何获取已设置的描述符呢?Object.getOwnPropertyDescriptor 可以完成此项工作。

复制代码 代码如下:

var obj = { };
Object.defineProperty(obj, 'attr', {
    value: 1,
    writable: false,
    configurable: true
});
var desc = Object.getOwnPropertyDescriptor(obj, 'attr');
console.dir(desc);

对象控制

前面说的 Object.defineProperty ,其操作的是对象的属性,而下面说的三个方法则直接操作对象。

Object.preventExtensions 可以使对象无法拥有新的属性:

复制代码 代码如下:

var obj = { };
obj.attr = 1;
Object.preventExtensions(obj);
obj.attr2 = 2; //fail

Object.seal 可以使对象仅剩属性值可以修改(如果属性为只读,则连属性值都无法修改):

复制代码 代码如下:

var obj = { };
obj.attr = 1;
Object.seal(obj);
obj.attr = 1.5;
delete obj.attr; // fail

Object.freeze 可以使对象完全无法被修改:

复制代码 代码如下:

var obj = { };
obj.attr = 1;
Object.freeze(obj);
obj.attr = 1.5; // fail
obj.attr2 = 2; //fail

然后大家可能又会问,怎么知道某个对象是否曾经被preventExtensions、seal或者freeze呢?答案就是分别调用 Object.isExtensible 、 Object.isSealed 、 Object.isFrozen ,这三个函数的用法比较简单,就不再累赘了。

总的来说,通过属性描述符可以进一步严格控制对象,加强程序逻辑的严谨性,唯一不足的就是,ES5在IE9里面才基本实现(IE9还不支持严格模式),考虑到国内IE8份额还比较高的情况,这套东西目前只能在移动端浏览器和Node.js里面用了。

时间: 2024-08-02 16:19:48

ECMAScript 5中的属性描述符详解_基础知识的相关文章

ECMAScript 5中的属性描述符详解

 这篇文章主要介绍了ECMAScript 5中的属性描述符详解,本文讲解了Object.defineProperty.数据描述符.存取描述符.获取属性描述符.对象控制等内容,需要的朋友可以参考下     属性描述符是ES5中新增的概念,其作用是给对象的属性增加更多的控制. Object.defineProperty 要研究属性描述符,首先要谈谈 Object.defineProperty 方法.这个方法的作用是给对象定义新属性或修改已存在的属性.其原型如下: 代码如下: Object.defin

JavaScript中的this关键字使用详解_基础知识

 和其它许多面向对象的语言一样,JavaScript 中也有 this 关键字,this 在函数中用来指向调用此方法的对象.实际编程中要判断 this 到底指向谁,一般可遵循以下原则:     如果该函数被 Function.call 或者 Function.apply 调用,那么 this 指向 call/apply 的第一个参数,如果参数是 null 或者 undefined,this 则指向全局对象(在浏览器中的话,全局对象就是 window 对象).     如果该函数被 Functio

JavaScript中的replace()方法使用详解_基础知识

 该方法找到一个正则表达式的字符串之间的匹配,并取代了匹配的子带的新的子串. 替换字符串可以包含以下特殊替换模式:  语法 string.replace(regexp/substr, newSubStr/function[, flags]); 下面是参数的详细信息:     regexp : 一个RegExp对象.匹配被替换参数的返回#2.     substr : 一个字符串,由newSubStr 来替换     newSubStr : 它取代从参数中收到的子字符串 #1.     funct

JavaScript中for循环的使用详解_基础知识

 我们已经看到,while循环有不同变种.本章将介绍另一种流行的循环叫做for循环. for 循环 for循环是循环最紧凑的形式,并包含有以下三个重要部分组成:     循环初始化计数器的初始值.初始化语句执行循环开始之前.     测试语句,将测试如果给定的条件是真还是假.如果条件为真,那么将要执行的循环中给定的代码,否则循环会退出来.     循环语句,可以增加或减少计数器. 可以把所有的三个部分中的一行用分号隔开. 语法 for (initialization; test conditio

JavaScript中getUTCSeconds()方法的使用详解_基础知识

 javascript Date.getUTCSeconds()方法返回按照通用时间在指定日期的秒.通过getUTCSeconds返回的值是0到59之间的一个整数.语法 Date.getUTCSeconds() 下面是参数的详细信息:     NA 返回值: 按照通用时间返回在指定的日期的秒数.例子: 下面的例子中打印当前时间给变量hrs的秒数部分. <html> <head> <title>JavaScript getUTCSeconds Method</tit

JavaScript中的getDay()方法使用详解_基础知识

 javascript Date.getDay()方法按照本地时间返回一周中的一天为所述指定的日期.通过getDay返回的值是对应于星期几的整数:0代表星期日,1代表星期一,2表示星期二,依此类推.语法 Date.getDay() 下面是参数的详细信息:     NA 返回值: 按照本地时间返回星期几为指定日期.例子: <html> <head> <title>JavaScript getDay Method</title> </head> &l

JavaScript中的some()方法使用详解_基础知识

 JavaScript数组some()方法测试数组中的某个元素是否通过由提供的功能来实现测试.语法 array.some(callback[, thisObject]); 下面是参数的详细信息:     callback : 函数用来测试每个元素.     thisObject : 对象作为该执行回调时使用. 返回值: 如果某些元素通过测试则返回true,否则为false.兼容性: 这个方法是一个JavaScript扩展到ECMA-262标准; 因此它可能不存在在标准的其他实现.为了使它工作,你

JavaScript中reduce()方法的使用详解_基础知识

 JavaScript 数组reduce()方法同时应用一个函数针对数组的两个值(从左到右),以减至一个值.语法 array.reduce(callback[, initialValue]); 下面是参数的详细信息:     callback : 函数执行在数组中每个值     initialValue : 对象作为第一个参数回调的第一次调用使用 返回值: 返回数组的减少单一个值兼容性: 这种方法是一个JavaScript扩展到ECMA-262标准; 因此它可能不存在在标准的其他实现.为了使它工

Javascript中的getUTCDay()方法使用详解_基础知识

 javascript Date.getUTCDay()方法按照通用时间在指定日期返回星期几.通过getUTCDay返回的值是对应于星期几的整数:0代表星期日,1代表星期一,2表示星期二,依此类推.语法 Date.getUTCDay() 下面是参数的详细信息:     NA 返回值: 按照通用时间返回指定日期的一周中的一天.例子: <html> <head> <title>JavaScript getUTCDay Method</title> </he