[译]因扩展Object.prototype而引发Object.defineProperty不可用的一个问题
by 紫云飞
at 2012-12-03 21:52:00
original http://www.cnblogs.com/ziyunfei/archive/2012/12/03/2799984.html
原文:http://d.hatena.ne.jp/teramako/20121129/p1
从ES-Discuss邮件列表中看到的.
var o = {};
Object.defineProperty(o, "hoge", { value: "OK" });
// TypeError: property descriptors must not specify a value or be writable when a getter or setter has been specified
问题描述就是:如果在Object.prototype上添加了名为
get
,set之类的属性的话,再执行
Object.defineProperty()的时候就很有可能发生问题.为什么?
不应该在Object.prototype上添加的属性
下面给出的属性名最不应该添加在Object.prototype上
- get
- set
- value
- writable
为什么?
Object.defineProperty的第三个参数Descriptor是个对象,指定了所定义属性的属性描述符.属性描述符一共有两种.
数据属性描述符(DataDescriptor)和访问器属性描述(AccessorDescriptor).
数据属性描述符对象上不能有get
,set属性,
访问器属性描述符对象上不能有value
,writable属性.
执行defineProperty的时候,在判断属性描述符对象中某个属性是否存在时使用的内部方法是[[HasProperty]].[[HasProperty]]会在[[Prototype]]上寻找属性,也就找到了Object.prototype上定义的那些属性.如果使用[[GetOwnProperty]]来判断的话就不会有这样的问题了.
上例中的问题就是,get和value两个属性不能同时存在,否则会报错.
解决办法
创建一个没有原型的属性描述符对象:
des.value = 123;
Object.defineProperty(obj,"key",des);
也可以使用非标准的魔法属性__proto__:
__proto__: null,
value: 123
});
同样,Object.defineProperties()以及
Object.create()的第二个参数中也有属性描述符对象,也有可能出现同样的问题
.