ECMAScript 5中的属性描述符详解


Posted in Javascript onMarch 02, 2015

属性描述符是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里面用了。

Javascript 相关文章推荐
Javascript 继承实现例子
Aug 12 Javascript
扩展javascript的Date方法实现代码(prototype)
Nov 20 Javascript
JavaScript Title、alt提示(Tips)实现源码解读
Dec 12 Javascript
Jquery uploadify图片上传插件无法上传的解决方法
Dec 16 Javascript
js中this用法实例详解
May 05 Javascript
实例讲解JavaScript的Backbone.js框架中的View视图
May 05 Javascript
JavaScript知识点总结(十六)之Javascript闭包(Closure)代码详解
May 31 Javascript
jquery实现垂直和水平菜单导航栏
Aug 27 Javascript
js实现无缝滚动图(可控制当前滚动的方向)
Feb 22 Javascript
基于Vue 2.0的模块化前端 UI 组件库小结
Dec 21 Javascript
JS实现带导航城市列表以及输入搜索功能
Jan 04 Javascript
layui 实现二级弹窗弹出之后 关闭一级弹窗的方法
Sep 18 Javascript
JS+CSS实现可以凹陷显示选中单元格的方法
Mar 02 #Javascript
JavaScript数组常用方法
Mar 02 #Javascript
使用npm发布Node.JS程序包教程
Mar 02 #Javascript
js实现点击链接后窗口缩小并居中的方法
Mar 02 #Javascript
运行Node.js的IIS扩展iisnode安装配置笔记
Mar 02 #Javascript
Javascript动画的实现原理浅析
Mar 02 #Javascript
JavaScript页面模板库handlebars的简单用法
Mar 02 #Javascript
You might like
PHP查看当前变量类型的方法
2015/07/31 PHP
Javascript读取cookie函数代码
2010/10/16 Javascript
jquery 插件学习(六)
2012/08/06 Javascript
JS 打印界面的CSS居中代码适用所有浏览器
2014/03/19 Javascript
JS实现的表格行上下移动操作示例
2016/08/03 Javascript
Javascript点击按钮随机改变数字与其颜色
2016/09/01 Javascript
Jquery Easyui日历组件Calender使用详解(23)
2016/12/18 Javascript
JavaScript BASE64算法实现(完美解决中文乱码)
2017/01/10 Javascript
vue元素实现动画过渡效果
2017/07/01 Javascript
详解node-ccap模块生成captcha验证码
2017/07/01 Javascript
vue select组件的使用与禁用实现代码
2018/04/10 Javascript
JS判断字符串是否为整数的方法--简单的正则判断
2018/07/23 Javascript
vue+element-ui集成随机验证码+用户名+密码的form表单验证功能
2018/08/05 Javascript
javascript关于“时间”的一次探索
2019/07/24 Javascript
uni-app实现点赞评论功能
2019/11/25 Javascript
js Math数学简单使用操作示例
2020/03/13 Javascript
Vue解决移动端弹窗滚动穿透问题
2020/12/15 Vue.js
[01:36]DOTA2完美大师赛趣味视频之与队友相处的十万个技巧
2017/11/19 DOTA
python使用paramiko模块实现ssh远程登陆上传文件并执行
2014/01/27 Python
Python logging模块用法示例
2018/08/28 Python
Python内置类型性能分析过程实例
2020/01/29 Python
解决reload(sys)后print失效的问题
2020/04/25 Python
python批量检查两个对应的txt文件的行数是否一致的实例代码
2020/10/31 Python
python 如何在测试中使用 Mock
2021/03/01 Python
FC-Moto美国:欧洲最大的摩托车服装和头盔商店之一
2019/08/24 全球购物
美国办公用品折扣网站:Shoplet.com
2019/11/24 全球购物
伦敦鲜花递送:Flower Station
2021/02/03 全球购物
const和static readonly区别
2013/05/20 面试题
办公室前台的岗位职责
2013/12/20 职场文书
财务出纳岗位职责
2014/02/03 职场文书
党校个人总结
2015/03/04 职场文书
2015年社区工会工作总结
2015/05/26 职场文书
python如何进行基准测试
2021/04/26 Python
详解Redis实现限流的三种方式
2021/04/27 Redis
浅谈Python数学建模之整数规划
2021/06/23 Python
python计算列表元素与乘积详情
2022/08/05 Python