我像这样扩展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.