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 相关文章推荐
使用jQuery和PHP实现类似360功能开关效果
Feb 12 Javascript
Jquery插件easyUi实现表单验证示例
Dec 15 Javascript
全面详细的jQuery常见开发技巧手册
Feb 21 Javascript
简单实现jQuery进度条轮播实例代码
Jun 20 Javascript
Angular2开发——组件规划篇
Mar 28 Javascript
react-router实现跳转传值的方法示例
May 27 Javascript
vue 2.0 购物车小球抛物线的示例代码
Feb 01 Javascript
vue2.0学习之axios的封装与vuex介绍
May 28 Javascript
javascript如何使用函数random来实现课堂随机点名方法详解
Jul 28 Javascript
JavaScript实现鼠标移入随机变换颜色
Nov 24 Javascript
如何制作自己的原生JavaScript路由
May 05 Javascript
vue实现省市区联动 element-china-area-data插件
Apr 22 Vue.js
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 生成的XML以FLASH获取为乱码终极解决
2009/08/07 PHP
phpmyadmin提示The mbstring extension is missing的解决方法
2014/12/17 PHP
php实现的顺序线性表示例
2019/05/04 PHP
mysqli扩展无法在PHP7下升级问题的解决
2019/09/10 PHP
jquery实用代码片段集合
2010/08/12 Javascript
Jquery常用技巧收集整理篇
2010/11/14 Javascript
学习面向对象之面向对象的基本概念:对象和其他基本要素
2010/11/30 Javascript
javascript-简单的计算器实现步骤分解(附图)
2013/05/30 Javascript
基于jquery的文章中所有图片width大小批量设置方法
2013/08/01 Javascript
JavaScript自定义方法实现trim()、Ltrim()、Rtrim()的功能
2013/11/03 Javascript
jquery实现图片灯箱明暗的遮罩效果
2013/11/15 Javascript
jQuery中fadeOut()方法用法实例
2014/12/24 Javascript
AngularJS内置指令
2015/02/04 Javascript
浅谈window对象的scrollBy()方法
2015/07/15 Javascript
用jQuery的AJax实现异步访问、异步加载
2016/11/02 Javascript
JavaScript编写九九乘法表(两种任选)
2017/02/04 Javascript
移动端使用localResizeIMG4压缩图片
2017/04/22 Javascript
详解Angular5路由传值方式及其相关问题
2018/04/28 Javascript
jQuery实现表单动态添加数据并提交的方法
2018/07/19 jQuery
微信小程序云开发 生成带参小程序码流程
2019/05/18 Javascript
JavaScript代码实现微博批量取消关注功能
2021/02/05 Javascript
利用Python爬取可用的代理IP
2016/08/18 Python
python smtplib模块自动收发邮件功能(一)
2018/05/22 Python
Python爬取qq空间说说的实例代码
2018/08/17 Python
在python3中pyqt5和mayavi不兼容问题的解决方法
2019/01/08 Python
django 外键创建注意事项说明
2020/05/20 Python
HTML5 Canvas+JS控制电脑或手机上的摄像头实例
2014/05/03 HTML / CSS
激光脱毛、蓝光和护肤:Tria Beauty
2019/03/28 全球购物
Timberland俄罗斯官方网上商店:全球领先的户外品牌
2020/03/15 全球购物
房产委托公证书样本
2014/04/04 职场文书
《天游峰的扫路人》教学反思
2014/04/25 职场文书
地震捐款倡议书
2014/08/29 职场文书
党员群众路线自我剖析材料
2014/10/06 职场文书
2014年环境整治工作总结
2014/12/10 职场文书
大学生个人简历自荐信
2015/03/06 职场文书
python使用torch随机初始化参数
2022/03/22 Python