继承方法到除原语以外的所有对象

Inherit Method to All Objects Except Primitives

本文关键字:对象 方法 原语 继承      更新时间:2023-09-26

我想添加一个能够在所有对象上工作的方法。最好它不存在于基本类型(数字、布尔值、字符串、符号)上,否则它无法工作。

这是可能的吗?如果是,怎么做?

使用对象。prototype不能工作,因为基本类型继承自Object.

您可以在Object的原型上设置函数,然后在不需要它的构造函数的原型上设置相应的成员undefined:

Object.prototype.foo = function() { alert("foo"); };
Number.prototype.foo = undefined;
var x = {};
var y = 10;
x.foo(); // works
y.foo(); // TypeError: y.foo is not a function

我不知道你为什么这样做,这看起来像一个代码气味,但你可以。

原语不是对象,不能从任何地方继承。它们可以封装在从Object.prototype继承的对象中,但这些都是对象。

它可能看起来像原语继承属性,但在底层,这些是在对象包装器上访问的,而不是原语。当GetValue解析属性引用时,

  • 如果HasPrimitiveBase(V)为true,则
    • Assert:在这种情况下,base将永远不会为空或未定义。
    • 基础ToObject ()。
  • 返回基地。[[有]](GetReferencedName (V ), GetThisValue (V ))。

因此,如果引用的基类是原始类型,则在尝试获取该属性之前将其转换为对象。

但是正如@nnnnnn所指出的,当您调用该方法时,this值将是原语,因为引用的基没有改变。注意,在草率模式下,它将被强制为一个对象。

实际上,如果您想要在调用方法时防止原语被用作引用的基础,您可以利用这一点并测试this值是否为对象:

Object.prototype.method = function() {
  "use strict";
  if(Object(this) !== this) {
    throw Error('This method can only be called on objects');
  }
  // ...
  return "works";
};
true.method(); // Error: This method can only be called on objects
new Boolean(true).method(); // "works"

从原语中删除方法的简化方法。

Object.prototype.test = function() { }
for(let { prototype } of [ String, Number, Boolean, Symbol ]) {
 prototype.test = undefined
}