解析JavaScript中instanceof对于不同的构造器或许都返回true


Posted in Javascript onDecember 03, 2013

我们知道 instanceof 运算符用来检查对象是否为某构造器的实例。下面列举它返回true的各种情景。

1、对象obj是通过new Constructor创建的,那么 obj instanceof Constructor 为true

function Person(n, a) { 
    this.name = n; 
    this.age = a; 
} 
var p = new Person('John Backus', 82); 
console.log(p instanceof Person); // true

2、如果存在继承关系,那么 子类实例 instanceof 父类 也会返回true
function A(){} 
function B(){} 
B.prototype = new A(); // B继承于A var b = new B(); 
console.log(b instanceof A); // true

3、由于Object是根类,所有其它自定义类都继承于它,因此 任意构造器的实例 instanceof Object 都返回true
function A() {} 
var a = new A(); 
console.log(a instanceof Object); // true var str = new String('hello'); 
console.log(str instanceof Object); // true 
var num = new Number(1); 
console.log(num instanceof Object); // true

甚至包括构造器自身
function A() {} 
console.log(A instanceof Object); // true 
console.log(String instanceof Object); // true 
console.log(Number instanceof Object); // true

4、所有构造器 instanceof Function 返回true
function A() {} 
console.log(A instanceof Function); // true 
console.log(String instanceof Function); // true 
console.log(Number instanceof Function); // true

以上四点总结为一句话:如果某实例是通过某类或其子类的创建的,那么instanceof就返回true。或者说某构造函数的原型 存在与对象obj的内部原型链上,那么返回true。即instanceof的结果与构造器自身并无直接关系。这在许多语言中都是通用的。

Java中定义了一个类Person,实例p对于Person和Object都返回true

class Person { 
    public String name; 
    public int age; 
    Person (String n, int a) { 
        this.name = name; 
        this.age = a; 
    } 
    public static void main(String[] args) { 
        Person p = new Person("John Backus", 82); 
        System.out.println(p instanceof Person); // true 
        System.out.println(p instanceof Object); // true 
    } 
}

Java中如果存在继承关系,那么 子类实例 instanceof 父类 也返回true
// 父类 
class Person { 
    public String name; 
    public int age; 
    Person (String n, int a) { 
        name = name; 
        age = a; 
    } 
} 
// 子类 
public class Man extends Person{ 
    public String university; 
    Man(String n, int a, String s) { 
        super(n, a); 
        university = s; 
    } 
    public static void main(String[] args) { 
        Man mm = new Man("John Resig", 29, "PKU"); 
        System.out.println(mm instanceof Man); // true 
        System.out.println(mm instanceof Person); // 也是true 
    } 
}

知道了这些,JS中以下的表现就不奇怪了
// 定义两个构造器 
function A(){} 
function B(){} 
A.prototype = B.prototype = {a: 1}; // 分别创建两个不同构造器的实例 
var a = new A(); 
var b = new B(); 
console.log(a instanceof B); // true 
console.log(b instanceof A); // true

我们看到a, b分别是用A和B创建的,但a instanceof B和 b instanceof A都是true。即a虽然不是用构造器B创建的,但仍然返回true。因为B.prototype存在于a的内部原型链上。

由于JS的动态语言特性,可以在运行时修改原型,因此下面返回false也不足为奇了。因为A.prototype已经不在a的内部原型链中,链条被打断了。

function A(){} 
var a = new A(); 
A.prototype = {}; // 动态修改原型,注意必须在创建a后 
console.log(a instanceof A); // false

注意这么写也打破了上面总结的第一条:对象obj是通过new Constructor创建的,那么obj instanceof Constructor 为true

实际在ECMAScript标准中(以5.1为准),instanceof 内部实现会调用构造器的内部方法[[HasInstance]],描述如下

解析JavaScript中instanceof对于不同的构造器或许都返回true

假如F是一个函数对象,当F(V)执行时,以下步骤将发生:

1、如果instanceof左运算元V不是对象类型,直接返回false

var a, b = 1, c = true, d = 'hello'; 
console.log(a instanceof Object); // false 这里a值为undefined 
console.log(b instanceof Object); // false 
console.log(c instanceof Object); // false 
console.log(d instanceof Object); // false

2/3、取构造器F的prototype属性,如果不是对象类型,须抛出TypeError异常,
function A(){} 
A.prototype = 1; // A的prototype设为非对象类型 
var a = new A(); 
console.log(a instanceof A);

各浏览器抛出的异常提示不同,

Firefox18:

解析JavaScript中instanceof对于不同的构造器或许都返回true

Chrome24:

解析JavaScript中instanceof对于不同的构造器或许都返回true

Safari6:

解析JavaScript中instanceof对于不同的构造器或许都返回true

Opera12:

解析JavaScript中instanceof对于不同的构造器或许都返回true

IE10:

解析JavaScript中instanceof对于不同的构造器或许都返回true

4、不断的执行以下逻辑:将V设为内部原型的V,如果V是null则返回false,如果V和O都指向同一个对象,则返回true。

Javascript 相关文章推荐
javascript中in运算符用法分析
Apr 28 Javascript
javascript连续赋值问题
Jul 08 Javascript
JavaScript的函数式编程基础指南
Mar 19 Javascript
Jquery uploadify 多余的Get请求(404错误)的解决方法
Jan 26 Javascript
关于js中的鼠标事件总结
Jul 11 Javascript
对vue v-if v-else-if v-else 的简单使用详解
Sep 29 Javascript
vue项目移动端实现ip输入框问题
Mar 19 Javascript
vue-mugen-scroll组件实现pc端滚动刷新
Aug 16 Javascript
JavaScript获取某一天所在的星期
Sep 05 Javascript
在vue中使用jsx语法的使用方法
Sep 30 Javascript
JS实现关闭小广告特效
Jan 29 Javascript
JS实现音量控制拖动
Jan 15 Javascript
探讨JavaScript中声明全局变量三种方式的异同
Dec 03 #Javascript
解析JavaScript中delete操作符不能删除的对象
Dec 03 #Javascript
解析Javascript小括号“()”的多义性
Dec 03 #Javascript
解析Javascript中中括号“[]”的多义性
Dec 03 #Javascript
jquery将一个表单序列化为一个对象的方法
Dec 02 #Javascript
jQuery获得内容和属性方法及示例
Dec 02 #Javascript
jquery如何实现锚点链接之间的平滑滚动
Dec 02 #Javascript
You might like
PHP最常用的ini函数分析 针对PHP.ini配置文件
2010/04/22 PHP
php修改文件上传限制方法汇总
2015/04/07 PHP
PHP连接MYSQL数据库的3种常用方法
2017/02/27 PHP
解决laravel 出现ajax请求419(unknown status)的问题
2019/09/03 PHP
laravel框架实现敏感词汇过滤功能示例
2020/02/15 PHP
jquery 多级下拉菜单核心代码
2010/05/21 Javascript
基于jquery完美拖拽,可返回拖动轨迹
2012/03/29 Javascript
很好用的js日历算法详细代码
2013/03/07 Javascript
利用js的Node遍历找到repeater的一个字段实例介绍
2013/04/25 Javascript
JavaScript数组Array对象增加和删除元素方法总结
2015/01/20 Javascript
JavaScript的MVVM库Vue.js入门学习笔记
2016/05/03 Javascript
javascript之Boolean类型对象
2016/06/07 Javascript
CSS3+JavaScript实现翻页幻灯片效果
2017/06/28 Javascript
实现图片首尾平滑轮播(JS原生方法—节流)
2017/10/17 Javascript
JS控制鼠标拒绝点击某一按钮的实例
2017/12/29 Javascript
Angular整合zTree的示例代码
2018/01/24 Javascript
nodejs微信扫码支付功能实现
2018/02/17 NodeJs
layui使用button按钮 点击出现弹层 弹层中加载表单的实例
2019/09/04 Javascript
layui实现二维码弹窗、并下载到本地的方法
2019/09/25 Javascript
jQuery实现朋友圈查看图片
2020/09/11 jQuery
Python实现的数据结构与算法之快速排序详解
2015/04/22 Python
Python求导数的方法
2015/05/09 Python
Python线程指南详细介绍
2017/01/05 Python
Python入门_条件控制(详解)
2017/05/16 Python
Django rest framework基本介绍与代码示例
2018/01/26 Python
Python综合应用名片管理系统案例详解
2020/01/03 Python
Python内置数据类型list各方法的性能测试过程解析
2020/01/07 Python
Python 支持向量机分类器的实现
2020/01/15 Python
Python函数参数定义及传递方式解析
2020/06/10 Python
详解python tkinter包获取本地绝对路径(以获取图片并展示)
2020/09/04 Python
火灾现场处置方案
2014/05/28 职场文书
何玥事迹观后感
2015/06/16 职场文书
廉洁自律心得体会2016
2016/01/13 职场文书
试了下Golang实现try catch的方法
2021/07/01 Golang
python利用while求100内的整数和方式
2021/11/07 Python
python中的getter与setter你了解吗
2022/03/24 Python