我像这样扩展Object:
Object.prototype.is_a = function (x) { return this instanceof x; }
所有的工作如预期
"foo".is_a(String) // true "foo".is_a(Object) // true "foo".is_a(Array) // false "foo".is_a(Function) // false "foo".is_a(Boolean) // false "foo".is_a(Date) // false "foo".is_a(Number) // false "foo".is_a(RegExp) // false
但当
"foo" instanceof String // false
这个关键字在is_a()函数上是否与foo相同?为什么会返回不同的结果?
解决方法
首先,您不能将instanceof应用于原始值,这就是为什么
"foo" instanceof String
返回false.原语不是对象,因此不能是构造函数的实例. *
那么为什么它似乎在is_a方法中工作?
在非严格模式下,该值在函数is always going to be an object中(步骤3).如果该值不是对象,则将其隐式转换为一个对象.你可以用console.log(typeof this)来测试.
这意味着字符串原语“foo”被转换为String对象new String(“foo”),这就是为什么可以在其上使用instanceof的原因.
在严格模式下,此值不一定是对象,不会自动转换(step 1).在这种情况下,您的方法将失败:
> Object.prototype.is_a = function (x) { 'use strict'; return this instanceof x; } > "foo".is_a(String) false
*:这是一个很简单的解释.实际上,instanceof运算符将评估委托给构造函数的内部[[HasInstance]]
method,该值被定义为如果传递的值不是对象则返回false.