为什么是 Object.defineProperty() 而不是 this.defineProperty() (对于对象)?

IT技术 javascript ecmascript-5
2021-01-17 20:34:35

我正在处理一个 JavaScript 项目,只是想知道为什么对象实例不继承defineProperty()和其他方法,而不必调用超类(超对象?)Object方法。

我查看了MDN 文档,实际上有“非标准”属性方法。

但这些已被弃用。为什么要转向Object方法?

在我看来,likeinstance.defineProperty(...)Object.defineProperty(instance, ...). 我也会对其他一些 Object 方法说同样的话。

3个回答

这是为了避免冲突 - 通常,对象的问题不具有您期望的值的属性。
JS 中的对象通常用作键值映射,键可以是任意字符串 - 例如__defineGetter__hasOwnProperty或者不那么特殊的东西。现在,当您想在未知对象上调用这样的函数时 - 就像hasOwnProperty在通用枚举函数中经常使用的那样,其中可能传入任何 JSON - 您永远无法确定是否获得了被覆盖的属性(甚至可能不是一个函数) ) 或您想要的原始文件,或者该对象是否完全继承了该属性。为了避免这个问题(或者这个 IE 错误),你必须使用Object.prototype.hasOwnProperty.call- 这很难看。

因此,将所有这些函数命名空间Object只是有用的,它是一个更清晰的 API,它将反射方法与对象的应用程序接口分开。这也有助于优化(简化静态分析)并使限制对沙箱中反射 API 的访问变得更容易——至少这是设计理念

您可能很高兴defineProperty在原型中有一个周围,​​但您只能在处理已知对象时安全地使用它。如果你仍然想要它(你知道什么时候用,什么时候不用),你可以使用

Object.defineProperty(Object.prototype, "defineProperty", {
    writable: true,
    enumberable: false,
    value: function(prop, descr) {
        return Object.defineProperty(this, prop, descr); 
    }
});
@BenjaminGruenbaum web.archive.org/web/20141214082618/http: //wiki.ecmascript.org/...
2021-03-13 20:34:35
实际上,真正的原因是沙盒和更好的 API - 我可以证明这一点,但是“rationale_for_es3_1_static_object_methodsaug26.pdf”已经死了。
2021-03-16 20:34:35
@BenjaminGruenbaum 我更新了答案,但不同意沙箱的原因。我认为这是对使用参数Meta对象-当你写一个沙箱这不是简单的限制访问Object.defineProperty ,并 {}.constructor.defineProperty比只限制访问{}.defineProperty
2021-03-18 20:34:35
还值得注意的是,对象不必从Object.prototypeES5继承它们可以通过 无从继承Object.create(null),也可以从继承自 的对象继承null这是非常有用的行为,但如果对象不继承自Object.prototype,则不会在 上定义方法Object.prototype任何想要defineProperty通用的都必须这样做var defineProperty = Function.prototype.call.bind(Object.prototype.defineProperty);,因此将其定义为实用程序函数而不是方法是有意义的。
2021-04-02 20:34:35
@BenjaminGruenbaum“更好的 API”有点模棱两可——我认为它更好,因为它避免了冲突。你说的沙箱是什么意思?
2021-04-08 20:34:35

这样做是为了避免冲突 - 请记住,每个方法Object.prototype也是每个用户定义对象中的方法。

想象一下你想要一个自定义方法的对象defineProperty——当Object.defineProperty它在它的原型上时会完全破坏事物

有趣的。到目前为止,我想出的唯一原因是人们喜欢重写原型,并且像这样“隐藏”这种方法可能会帮助您避免一些错误。特别是因为好的方法名称,因为它比例如__defineGetter__.

似乎很多特性都依赖于这个功能(链接),所以在这种情况下让它更加全球化和安全是有意义的。