JavaScript操作符instanceof揭秘

开发 前端
在JavaScript中,我们可以用instanceof操作符来判断对象是否是某个类的实例,如果obj instaceof Class返回true,那么我们认为obj是Class的实例,obj要么由Class创建,要么由Class的子类创建。

在JavaScript中,我们可以用instanceof操作符来判断对象是否是某个类的实例,如果obj instaceof Class返回true,那么我们认为obj是Class的实例,obj要么由Class创建,要么由Class的子类创建。来自Java或其他强类型语言的开发者一定认为如果obj instaceof Class返回true,那么obj肯定拥有Class的所有属性。事实是这样么?我们看下面的代码:(斑头雁原创:http://bantouyan.iteye.com)

Js代码

  1. function ClassA()     
  2. {     
  3.     this.ma = 'ClassA';     
  4. }     
  5. ClassA.prototype.fa = function(){return 'prototype function';};     
  6.     
  7. function ClassB()     
  8. {     
  9.     this.mb = 'ClassB';     
  10. }     
  11. ClassB.prototype = ClassA.prototype;     
  12.     
  13. var obja = new ClassA();     
  14. alert(('ma' in obja) + ' ' + obja.hasOwnProperty('ma')); //output true true     
  15. alert(('mb' in obja) + ' ' + obja.hasOwnProperty('mb')); //output false false     
  16. alert(('fa' in obja) + ' ' + obja.hasOwnProperty('fa')); //output true false     
  17. alert(('fb' in obja) + ' ' + obja.hasOwnProperty('fb')); //output true false     
  18. alert(obja instanceof ClassA);   //output true     
  19. alert(obja instanceof ClassB);   //output true     
  20.     
  21. var objb = new ClassB();     
  22. alert(('ma' in objb) + ' ' + objb.hasOwnProperty('ma')); //output false false     
  23. alert(('mb' in objb) + ' ' + objb.hasOwnProperty('mb')); //output true true     
  24. alert(('fa' in objb) + ' ' + objb.hasOwnProperty('fa')); //output true false     
  25. alert(('fb' in objb) + ' ' + objb.hasOwnProperty('fb')); //output true false     
  26. alert(objb instanceof ClassA);   //output true     
  27. alert(objb instanceof ClassB);   //output true    

 

在这段代码中,我们定义了两个类,ClassA与ClassB,还给ClassA创建了原型方法fa,给ClassB创建了原型方法fb,并分别创建了对象obja与objb。直觉上,我们并不认为ClassA与ClassB是同一个类,所以obja不是ClassB的实例,objb也不是ClassA的实例,而且obja不会拥有属性fb,objb也不会拥有属性fa,但是运行结果告诉我们,JavaScript并不这样认为。obja与objb都拥有属性fa与fb,它们既是ClassA的实例也是ClassB的实例。下面我们分析一下原因。(斑头雁原创:http://bantouyan.iteye.com)

ClassA的prototype与ClassB的prototype是同一个对象,所以给ClassA增加原型方法fa会影响到ClassB,给ClassB增加的原型方法fb也会影响到ClassA,所以obja与objb都拥有属性fa与fb,这一点也比较容易理解。oba没有ClassB的实例属性mb但却是ClassB的实例,objb也没有ClassA的实例属性ma但却是ClassA的实例,这说明instanceof的判断与实例属性无关。既然instanceof与实例属性无关,那么它就跟原型属性有关。事实上,JavaScript根据原型判定instanceof的运算结果。(斑头雁原创:http://bantouyan.iteye.com)

我们知道同一个构造函数所创建的对象与这个构造函数共享同一个原型(只不一般不能直接访问过对象的原型),而ClassA与ClassB也共享同一个原型,那么obja与objb也共享同一个原型,所以可以这样认为,如果对象与类共享一个原型,那么对象就是这个类的实例,instanceof运算就返回true。下面我们看一下继承的情况。(斑头雁原创:http://bantouyan.iteye.com)

Js代码

  1. function ClassA()     
  2. {     
  3.     this.ma = 'ClassA';     
  4. }     
  5. ClassA.prototype.fa = function(){return 'prototype function fa';};     
  6.     
  7. function ClassB()     
  8. {     
  9.     this.mb = 'ClassB';     
  10. }     
  11. ClassB.prototype = ClassA.prototype;     
  12. ClassB.prototype.fb = function(){return 'prototype function fb';};     
  13.     
  14. function ClassC()     
  15. {     
  16.     this.mc = 'ClassC';     
  17. }     
  18. ClassC.prototype = new ClassB();     
  19. ClassC.prototype.fc = function(){return 'prototype function fc';};     
  20.     
  21. var objc = new ClassC();     
  22. alert(objc instanceof ClassB);   //output true     
  23. alert(objc instanceof ClassA);   //output true     
  24. alert(('ma' in objc) + ' ' + objc.hasOwnProperty('ma')); //output false false     
  25. alert(('mb' in objc) + ' ' + objc.hasOwnProperty('mb')); //output true false     
  26. alert(('mc' in objc) + ' ' + objc.hasOwnProperty('mc')); //output true true     
  27. alert(('fa' in objc) + ' ' + objc.hasOwnProperty('fa')); //output true false     
  28. alert(('fb' in objc) + ' ' + objc.hasOwnProperty('fb')); //output true false     
  29. alert(('fc' in objc) + ' ' + objc.hasOwnProperty('fc')); //output true false    

 

ClassC采用原型链方法继承了ClassB,所以ClassC的对象objc是ClassB的实例,但是运行结果告诉我们,objc也是ClassA的实例。objc是ClassC的实例,所以objc与ClassC共享一个原型。ClassC的原型是ClassB的实例,所以ClassC的原型的原型与ClassB的原型是同一个原型,而ClassB与ClassA共享同一个原型,所以ClassC的原型的原型与ClassA的原型是同一个原型,即objc的原型的原型与ClassA的原型是同一个原型。由此看来,instanceof可以根据对象原型的原型,即原型链上的原型,判定运算的结果,即如果类的原型与对象原型链上的某一个原型是同一个对象,那么instanceof运算将返回true,否则返回false。下面我们用一段更长的代码来验证这个结论。(斑头雁原创:http://bantouyan.iteye.com)

Js代码

  1. function ClassA()     
  2. {     
  3.     this.ma = 'ClassA';     
  4. }     
  5. ClassA.prototype.fa = function(){return 'prototype function fa';};     
  6.     
  7. function ClassB()     
  8. {     
  9.     this.mb = 'ClassB';     
  10. }     
  11. ClassB.prototype = ClassA.prototype;     
  12. ClassB.prototype.fb = function(){return 'prototype function fb';};     
  13.     
  14. function ClassC()     
  15. {     
  16.     this.mc = 'ClassC';     
  17. }     
  18. ClassC.prototype = new ClassB();     
  19. ClassC.prototype.fc = function(){return 'prototype function fc';};     
  20.     
  21. function ClassD()     
  22. {     
  23.     this.md = 'ClassD';     
  24. }     
  25. ClassD.prototype = ClassC.prototype;     
  26. ClassC.prototype.fd = function(){return 'prototype function fd';};     
  27.     
  28. function ClassE()     
  29. {     
  30.     this.me = 'ClassE';     
  31. }     
  32. ClassE.prototype = new ClassD();     
  33. ClassE.prototype.fe = function(){return 'prototype function fe';};     
  34.     
  35. var obje = new ClassE();     
  36. alert(obje instanceof ClassA);   //output true     
  37. alert(obje instanceof ClassB);   //output true     
  38. alert(obje instanceof ClassC);   //output true     
  39. alert(obje instanceof ClassD);   //output true     
  40. alert(obje instanceof ClassE);   //output true     
  41.     
  42. alert(('ma' in obje) + ' ' + obje.hasOwnProperty('ma')); //output false false     
  43. alert(('mb' in obje) + ' ' + obje.hasOwnProperty('mb')); //output true false     
  44. alert(('mc' in obje) + ' ' + obje.hasOwnProperty('mc')); //output false false     
  45. alert(('md' in obje) + ' ' + obje.hasOwnProperty('md')); //output true false     
  46. alert(('me' in obje) + ' ' + obje.hasOwnProperty('me')); //output true true     
  47.     
  48. alert(('fa' in obje) + ' ' + obje.hasOwnProperty('fa')); //output true false     
  49. alert(('fb' in obje) + ' ' + obje.hasOwnProperty('fb')); //output true false     
  50. alert(('fc' in obje) + ' ' + obje.hasOwnProperty('fc')); //output true false     
  51. alert(('fd' in obje) + ' ' + obje.hasOwnProperty('fd')); //output true false     
  52. alert(('fe' in obje) + ' ' + obje.hasOwnProperty('fe')); //output true false    

 

ClassA的原型与ClassB的原型共享同一个对象,ClassC的原型与ClassD的原型也共享一个对象,而ClassC是ClassB的子类,ClassE是ClassD的子类,而obje是ClassE的实例。所以obje的原型链的***个原型是ClassE的原型,它是ClassD的实例,故第二个原型是ClassD的原型,也就是ClassC的原型,ClassC的原型是ClassB的实例,所以原型链上第三个原型是ClassB的原型,也就是ClassA的原型。所以,ClassA、ClassB、ClassC、ClassD与ClassE的原型都在对象obje的原型链上,所以obje是这些类的实例。这也验证了前面的结论,即如果类的原型与对象原型链上的某一个原型是同一个对象,那么instanceof运算将返回true。虽然obje是ClassA、ClassC的实例,但是它并没有ClassA的实例属性ma,ClassC的实例属性mc。(斑头雁原创:http://bantouyan.iteye.com)

综合前面的论述与验证,我们可以得出结论,如果obj instanceof Class返回true,那么Class的原型与obj原型链上的某个原型是同一个对象,但这并不意味着obj拥有Class的所有实例属性(但肯定拥有Class的所有原型属性)。

【编辑推荐】

  1. 10个超棒的HTML 5素描及绘画设计工具
  2. 学习HTML 5十佳站点推荐
  3. 12个精妙有趣的HTML 5应用
  4. 10个令人惊奇的HTML5和JavaScript效果
  5. 技术大牛谈HTML 5设计原理
责任编辑:金贺 来源: ITEYE博客
相关推荐

2010-07-14 14:55:07

Perl操作符

2009-08-19 17:26:28

C# 操作符

2021-10-31 18:59:55

Python操作符用法

2009-07-21 09:31:00

Scala操作符

2012-02-06 09:13:23

LINQ

2010-07-14 14:18:51

Perl操作符

2009-09-15 17:16:58

LINQ查询操作符

2009-09-16 09:09:23

Linq Contai

2010-07-14 14:30:31

Perl操作符

2010-07-19 11:00:24

Perl操作符

2010-01-28 11:16:28

C++操作符

2022-10-08 07:49:55

New操作符函数

2009-11-27 09:41:56

LINQ

2017-01-03 15:56:20

RxJava操作符Android

2010-01-27 11:00:17

C++操作符

2016-12-28 09:48:09

AndroidRxJava操作符

2016-12-28 09:54:50

AndroidRxJava操作符

2009-11-30 16:48:08

PHP操作符

2009-07-14 18:34:22

Jython操作符重载

2017-01-03 16:12:13

RxJava操作符Android
点赞
收藏

51CTO技术栈公众号