我有一种情况需要检查构造函数(X)在其原型链中是否有另一个构造函数(Y)(或者是Y本身)。
执行此操作的最快方法可能是(new X()) instanceof Y
。在这种情况下,这不是一个选项,因为有问题的构造函数可能会在没有有效参数的情况下实例化。
我考虑的下一个方法是:
const doesInherit = (A, B) => {
while (A) {
if (A === B) return true;
A = Object.getPrototypeOf(A);
}
return false;
}
这是有效的,但是我无法摆脱这种感觉,即我错过了一些更简单的检查方法。有吗?
答案 0 :(得分:26)
由于instanceof
的工作方式,您应该可以
A.prototype instanceof B
但是这只会测试继承,你应该比较A === B
来测试自我引用:
A === B || A.prototype instanceof B
class A {}
class B extends A {}
class C extends B {}
console.log(C === C) // true
console.log(C.prototype instanceof B) // true
console.log(C.prototype instanceof A) // true
instanceof
基本上实现如下:
function instanceof(obj, Constr) {
var proto;
while ((proto = Object.getProtoypeOf(obj)) {
if (proto === Constr.prototype) {
return true;
}
}
return false;
}
迭代对象的原型链,并检查是否有任何原型等于构造函数prototype
属性。
所以几乎就像你在做什么,但在内部。
答案 1 :(得分:13)
还有Object.prototype.isPrototypeOf()
。看起来像一个完美的用例,没有?
class A {}
class B extends A {}
class C extends B {}
console.log(C === C)
console.log(B.isPrototypeOf(C))
console.log(A.isPrototypeOf(C))
答案 2 :(得分:3)
注意:上面用isPrototypeOf()检查ES6类A,B,C继承链的答案很有用。但是它并没有像预期的ES6类对象,数组,函数等那样工作。
Object.prototype.isPrototypeOf (Array.prototype); // => true
可是:
Object.isPrototypeOf (Array); // => false
这应该是应该的。 Array的INSTANCES继承Object.prototype的方法。但是"班级"数组不会继承"类"宾语。如果向Object添加方法,则无法通过Array调用它。 Array不继承Object的方法(如果有的话)。只是用户创建的类的工作方式不同!
也许我们不应该将Object和Array和Function视为"类"完全 - 即使你可以创建它们的实例。他们只是"建设者"。或者我们可以说它们是类,但JavaScript中的内置类与用户创建的类不同。