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 相关文章推荐
JS Array对象入门分析
Oct 30 Javascript
js单独获取一个checkbox看其是否被选中
Sep 22 Javascript
18个非常棒的jQuery代码片段
Nov 02 Javascript
jQuery简单实现提交数据出现loading进度条的方法
Mar 29 Javascript
js中toString()和String()区别详解
Mar 23 Javascript
vue中mint-ui环境搭建详细介绍
Apr 06 Javascript
addEventListener()与removeEventListener()解析
Apr 20 Javascript
npm 更改默认全局路径以及国内镜像的方法
May 16 Javascript
Vue ElementUi同时校验多个表单(巧用new promise)
Jun 06 Javascript
微信小程序实现九宫格抽奖
Apr 15 Javascript
Javascript读写cookie的实例源码
Mar 16 Javascript
JavaScript的查询机制LHS和RHS解析
Aug 16 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 expects parameter 1 to be resource, array given 错误
2011/03/23 PHP
PHP中设置时区,记录日志文件的实现代码
2013/01/07 PHP
php笔记之:文章中图片处理的使用
2013/04/26 PHP
Zend Framework教程之前端控制器Zend_Controller_Front用法详解
2016/03/07 PHP
实例分析基于PHP微信网页获取用户信息
2017/11/24 PHP
[原创]图片分页查看
2006/08/28 Javascript
基于jquery的鼠标拖动效果代码
2012/05/30 Javascript
javascript 判断中文字符长度的函数代码
2012/08/27 Javascript
取得元素的左和上偏移量的方法
2014/09/17 Javascript
Bootstrap入门书籍之(零)Bootstrap简介
2016/02/17 Javascript
JQuery EasyUI的使用
2016/02/24 Javascript
深入理解jQuery之防止冒泡事件
2016/05/24 Javascript
详解微信小程序开发之——wx.showToast(OBJECT)的使用
2017/01/18 Javascript
如何使用CSS3和JQuery easing 插件制作绚丽菜单
2019/06/18 jQuery
es6中class类静态方法,静态属性,实例属性,实例方法的理解与应用分析
2020/02/15 Javascript
python进阶教程之循环对象
2014/08/30 Python
python在指定目录下查找gif文件的方法
2015/05/04 Python
Python 专题二 条件语句和循环语句的基础知识
2017/03/19 Python
利用Python yagmail三行代码实现发送邮件
2018/05/11 Python
详解Python最长公共子串和最长公共子序列的实现
2018/07/07 Python
Python爬虫框架Scrapy基本用法入门教程
2018/07/26 Python
Python subprocess库的使用详解
2018/10/26 Python
使用python模拟命令行终端的示例
2019/08/13 Python
Django项目使用ckeditor详解(不使用admin)
2019/12/17 Python
pytorch 归一化与反归一化实例
2019/12/31 Python
python isinstance函数用法详解
2020/02/13 Python
2021年的Python 时间轴和即将推出的功能详解
2020/07/27 Python
卡塔尔航空官方网站:Qatar Airways
2017/02/08 全球购物
超市端午节活动方案
2014/01/23 职场文书
小学教师自我剖析材料
2014/09/29 职场文书
2014年银行信贷员工作总结
2014/12/08 职场文书
大二学年个人总结
2015/03/03 职场文书
行政处罚听证告知书
2015/07/01 职场文书
一年之计:2019年下半年的计划
2019/05/07 职场文书
最新农村养殖致富:资金投入较低的创业项目有哪些?
2019/09/26 职场文书
详解thinkphp的Auth类认证
2021/05/28 PHP