JavaScript的instanceof运算符学习教程


Posted in Javascript onJune 08, 2016

语法

object instanceof constructor

参数
object:
要检测的对象.
constructor:
某个构造函数

描述:
instanceof 运算符用来检测 constructor.prototype 是否存在于参数 object 的原型链上。

// 定义构造函数
function C(){} 
function D(){} 

var o = new C();

// true,因为 Object.getPrototypeOf(o) === C.prototype
o instanceof C; 

// false,因为 D.prototype不在o的原型链上
o instanceof D; 

o instanceof Object; // true,因为Object.prototype.isPrototypeOf(o)返回true
C.prototype instanceof Object // true,同上

C.prototype = {};
var o2 = new C();

o2 instanceof C; // true

o instanceof C; // false,C.prototype指向了一个空对象,这个空对象不在o的原型链上.

D.prototype = new C(); // 继承
var o3 = new D();
o3 instanceof D; // true
o3 instanceof C; // true

需要注意的是,如果表达式 obj instanceof Foo 返回true,则并不意味着该表达式会永远返回ture,因为Foo.prototype属性的值有可能会改变,改变之后的值很有可能不存在于obj的原型链上,这时原表达式的值就会成为false。另外一种情况下,原表达式的值也会改变,就是改变对象obj的原型链的情况,虽然在目前的ES规范中,我们只能读取对象的原型而不能改变它,但借助于非标准的__proto__魔法属性,是可以实现的。比如执行obj.__proto__ = {}之后,obj instanceof Foo就会返回false了。

instanceof和多全局对象(多个frame或多个window之间的交互)

在浏览器中,我们的脚本可能需要在多个窗口之间进行交互。多个窗口意味着多个全局环境,不同的全局环境拥有不同的全局对象,从而拥有不同的内置类型构造函数。这可能会引发一些问题。比如,表达式 [] instanceof window.frames[0].Array 会返回false,因为 Array.prototype !== window.frames[0].Array.prototype,因此你必须使用 Array.isArray(myObj) 或者 Object.prototype.toString.call(myObj) === "[object Array]"来判断myObj是否是数组。

示例
instanceof的常规用法是判断a是否是b类型:

console.log(true instanceof Boolean); // false 
console.log(new Number(1) instanceof Number); // true

instanceof还能判断父类型:

function Father() {}
function Child() {}
Child.prototype = new Father();
var a = new Child();
console.log(a instanceof Child); // true
console.log(a instanceof Father); // true

Child构造函数继承自Father,实例a是Child构造的无疑,但是为何也是Father的实例呢?其实instanceof运算符的内核可以简单地用以下代码描述:

function check(a, b) {
 while(a.__proto__) {
  if(a.__proto__ === b.prototype)
   return true;
  a = a.__proto__;
 }
 return false;
}

function Foo() {}
console.log(Object instanceof Object === check(Object, Object)); // true 
console.log(Function instanceof Function === check(Function, Function)); // true 
console.log(Number instanceof Number === check(Number, Number)); // true 
console.log(String instanceof String === check(String, String)); // true 
console.log(Function instanceof Object === check(Function, Object)); // true 
console.log(Foo instanceof Function === check(Foo, Function)); // true 
console.log(Foo instanceof Foo === check(Foo, Foo)); // true

简单地说,a如果是b的实例,那么a肯定能使用b的prototype中定义的方法和属性,那么用代码表示就是a的原型链中有b.prototype取值相同的对象,于是顺着a的原型链一层层找就行了。

另外值得注意的是,String Number Boolean 以及Function等都是函数,而函数则是统一由Function构造而来的,so它们和任何单纯的函数一样,能用Function上的原型属性:

Function.prototype.a = 10;
console.log(String.a); // 10

最后来简单讲讲最开始的两道题吧。

// 为了方便表述,首先区分左侧表达式和右侧表达式
 FunctionL = Function, FunctionR = Function; 
 // 下面根据规范逐步推演
 O = FunctionR.prototype = Function.prototype 
 L = FunctionL.__proto__ = Function.prototype 
 // 第一次判断
 O == L 
 // 返回 true

// 为了方便表述,首先区分左侧表达式和右侧表达式
 StringL = String, StringR = String; 
 // 下面根据规范逐步推演
 O = StringR.prototype = String.prototype 
 L = StringL.__proto__ = Function.prototype 
 // 第一次判断
 O != L
 // 循环再次查找 L 是否还有 __proto__ 
 L = String.prototype.__proto__ = Object.prototype 
 // 第二次判断
 O != L 
 // 再次循环查找 L 是否还有 __proto__ 
 L = String.prototype.__proto__ = null 
 // 第三次判断
 L == null 
 // 返回 false
Javascript 相关文章推荐
Ext面向对象开发实践(续)
Nov 18 Javascript
使用jquery实现图文切换效果另加特效
Jan 20 Javascript
jQuery选择器源码解读(三):tokenize方法
Mar 31 Javascript
详解webpack介绍&安装&常用命令
Jun 29 Javascript
详谈表单重复提交的三种情况及解决方法
Aug 16 Javascript
Vue 进入/离开动画效果
Dec 26 Javascript
vuex操作state对象的实例代码
Apr 25 Javascript
vue指令之表单控件绑定v-model v-model与v-bind结合使用
Apr 17 Javascript
node实现mock-plugin中间件的方法
Dec 25 Javascript
vue2.0实现列表数据增加和删除
Jun 17 Javascript
vue-router之解决addRoutes使用遇到的坑
Jul 19 Javascript
html+vue.js 实现漂亮分页功能可兼容IE
Nov 07 Javascript
JavaScript中instanceof运算符的使用示例
Jun 08 #Javascript
实例讲解JavaScript中instanceof运算符的用法
Jun 08 #Javascript
js获取对象、数组的实际长度,元素实际个数的实现代码
Jun 08 #Javascript
JS & JQuery 动态添加 select option
Jun 08 #Javascript
Jquery ajax请求导出Excel表格的实现代码
Jun 08 #Javascript
浅谈几种常用的JS类定义方法
Jun 08 #Javascript
浅谈javascript中的constructor
Jun 08 #Javascript
You might like
有关phpmailer的详细介绍及使用方法
2013/01/28 PHP
php 地区分类排序算法
2013/07/01 PHP
ThinkPHP模板自定义标签使用方法
2014/06/26 PHP
PHP网站自动化配置的实现方法(必看)
2017/05/27 PHP
JavaScript获得选中文本内容的方法
2008/12/02 Javascript
Extjs4 GridPanel的主要配置参数详细介绍
2013/04/18 Javascript
一个JavaScript防止表单重复提交的实例
2014/10/21 Javascript
JS实现在网页中弹出一个输入框的方法
2015/03/03 Javascript
jquery 动态合并单元格的实现方法
2016/08/26 Javascript
JavaScript使用readAsDataUrl方法预览图片
2017/05/10 Javascript
React入门教程之Hello World以及环境搭建详解
2017/07/11 Javascript
JavaScript中click和onclick本质区别与用法分析
2018/06/07 Javascript
在 vue-cli v3.0 中使用 SCSS/SASS的方法
2018/06/14 Javascript
[50:45]2018DOTA2亚洲邀请赛 4.6 淘汰赛 VP vs TNC 第一场
2018/04/10 DOTA
[52:31]VP vs Serenity 2018国际邀请赛小组赛BO2 第二场 8.16
2018/08/17 DOTA
简单谈谈python中的多进程
2016/11/06 Python
Python使用正则表达式抓取网页图片的方法示例
2017/04/21 Python
解决python3中cv2读取中文路径的问题
2018/12/05 Python
Python使用matplotlib 模块scatter方法画散点图示例
2019/09/27 Python
Django实现auth模块下的登录注册与注销功能
2019/10/10 Python
如何解决django-celery启动后迅速关闭
2019/10/16 Python
使用Python实现 学生学籍管理系统
2019/11/26 Python
使用pyshp包进行shapefile文件修改的例子
2019/12/06 Python
python标准库sys和OS的函数使用方法与实例详解
2020/02/12 Python
波兰最大的宠物用品网上商店:FERA.PL
2019/08/11 全球购物
车间主管岗位职责
2013/11/14 职场文书
新学期红领巾广播稿
2014/01/14 职场文书
小学师德标兵先进事迹材料
2014/05/25 职场文书
教师作风建设剖析材料
2014/10/11 职场文书
2014幼儿园小班工作总结
2014/11/10 职场文书
装配车间主任岗位职责
2015/04/08 职场文书
小学作文之描写天气
2019/08/15 职场文书
再谈python_tkinter弹出对话框创建
2022/03/20 Python
vue3.0 数字翻牌组件的使用方法详解
2022/04/20 Vue.js
配置nginx负载均衡
2022/05/06 Servers
MySQL常用慢查询分析工具详解
2022/08/14 MySQL