理解js中的原型链

对象有”prototype”属性,函数对象有”prototype”属性,原型对象有”constructor”属性。

关于原型

  • 在JavaScript中,原型也是一个对象,通过原型可以实现对象的属性继承,JavaScript的对象实例中都包含了”[[Prototype]]”内部属性,这个属性所对应的就是该对象的原型。“[[Prototype]]”作为对象的内部属性,是不能被直接访问的。所以为了方便查看一个对象的原型,Firefox和Chrome中提供了__proto__这个非标准(不是所有浏览器都支持)的访问器。在JavaScript的原型对象还包含一个”constructor”属性,这个属性对应创建所有指向该原型的实例的构造函数。
  • 在JavaScript中,只要创建了一个新函数,那么这个函数就有一个prototype属性,这个属性指向函数的原型对象。在默认情况下所有原型都会自动获取一个constructor(构造函数)属性,这个属性包含一个指向prototype属性所在函数的指针。当一个函数被用作构造函数来创建实例时,这个函数的prototype属性值会被作为原型赋值给所有对象实例(也就是设置 实例的`__proto__`属性),也就是说,所有实例的原型引用的是函数的prototype属性。(****`只有函数对象才会有这个属性!`****)
  • JavaScript中函数也是对象,所以就可以通过_proto_查找到构造函数对象的原型。
    Function对象作为一个函数,就会有prototype属性,该属性将对应”function () {}”对象。
    Function对象作为一个对象,就有__proto__属性,该属性对应”Function.prototype”,也就是说,”Function._proto_ === Function.prototype”。
  • 对于所有的对象,都有__proto__属性,这个属性对应与对象的原型.
    对于函数对象,除了__proto__属性之外,还有prototype属性,当一个函数被用作构造函数来创建实例时,该函数的prototype属性值将被作为原型赋值给所有对象实例(也就是设置实例的__proto__属性)

原型链

因为每个对象和原型都有原型,对象的原型指向原型对象,而父的原型又指向父的父,这种原型层层连接起来的就构成了原型链。

主要思想: 利用prototype属性重写原型对象。利用原型让一个引用类型继承另一个引用类型的属性和方法。

 面向对象语言有两种继承方式:接口继承(只继承方法名);实现继承(继承实际的方法)。但在ECMAScript中,函数名没多大含义,只是函数体的引用而已,因此,ECMAScript无法实现接口继承,只支持实现继承。实现继承,主要是依靠原型链来完成的

构造函数、原型、实例之间的关系
(1)每个构造函数都有一个原型对象,Func.prototype指向了这个原型对象。
(2)原型对象都包含一个指向构造函数的指针,Func.prototype.constructor等于Func,可以理解为prototype和constructor是两个方向相反的指针,在构造函数和原型之间搭建起桥梁。
(3)每个实例,都包含一个指向原型对象的内部指针__proto__,当然这个对开发人员是不可见的。
(4)如果把构造函数A的原型,设成构造函数B的一个实例,那么构造函数A的原型里就包含了指向另一个原型的指针,从而形成了原型链。这样就实现了继承。

(5)通过原型链实现继承的情况下,搜索过程会沿着原型链向上:使用对象的属性或方法时,解析器先搜索实例内部,找不到就搜索实例的原型内部,再找不到就搜索实例的原型的构造函数的原型内部,一级一级向上查找,一直到原型链末端才会停下来。

(6)原型链的最顶端是Object,这就是为什么所有引用类型instanceof Object都会返回true。换句话说,只要instanceof后面的构造函数在实例的原型链中,都会返回true

如下:

function SuperType()
{
    this.property=true;
}
SuperType.prototype.getSuperValue=function(){
    return this.property;
};
function SubType()
{
    this.subProperty=false;
}
//继承SuperType
SubType.prototype=new SuperType();
SubType.prototype.getSubValue=function(){
    return this.subProperty;
}

var instance=new SubType();
alert(instance.getSuperValue());//true

 

原型链结构图如下:

 一个例子

 1 function Person (name) { this.name = name; }
 2 function Mother () { }
 3 Mother.prototype = {    //Mother的原型
 4     age: 18,
 5     home: ['Beijing', 'Shanghai']
 6 };
 7 Person.prototype = new Mother(); //Person的原型为Mother
 8 
 9 //用chrome调试工具查看,提供了__proto__接口查看原型
10 var p1 = new Person('Jack'); //p1:'Jack'; __proto__:{__proto__:18,['Beijing','Shanghai']}
11 var p2 = new Person('Mark'); //p2:'Mark'; __proto__:{__proto__:18,['Beijing','Shanghai']}
12 
13 p1.age = 20;  
14 /* 实例不能改变原型的基本值属性
15  * 在p1实例下增加一个age属性的普通操作,与原型无关。跟var o={}; o.age=20一样。
16  * p1:下面多了个属性age,而__proto__跟 Mother.prototype一样,age=18。
17  * p2:只有属性name,__proto__跟 Mother.prototype一样
18  */
19 
20 p1.home[0] = 'Shenzhen'; 
21 /* 原型中引用类型属性的共享
22  * p1:'Jack',20; __proto__:{__proto__:18,['Shenzhen','Shanghai']}
23  * p2:'Mark';    __proto__:{__proto__:18,['Shenzhen','Shanghai']}
24  */
25 
26 p1.home = ['Hangzhou', 'Guangzhou']; 
27 /* 其实跟p1.age=20一样的操作。换成这个理解: var o={}; o.home=['big','house']
28  * p1:'Jack',20,['Hangzhou','Guangzhou']; __proto__:{__proto__:18,['Shenzhen','Shanghai']}
29  * p2:'Mark';                             __proto__:{__proto__:18,['Shenzhen','Shanghai']}
30  */
31 
32 delete p1.age;    
33 /* 删除实例的属性之后,原本被覆盖的原型值就重见天日了。
34  * 这就是向上搜索机制
35  * p1:'Jack',['Hangzhou','Guangzhou']; __proto__:{__proto__:18,['Shenzhen','Shanghai']}
36  * p2:'Mark';                          __proto__:{__proto__:18,['Shenzhen','Shanghai']}
37  */
38 
39 Person.prototype.lastName = 'Jin'; 
40 /* 改写原型,动态反应到实例中。
41  * 注意,这里我们改写的是Person的原型,就是往Mother里加一个lastName属性,等同于Mother.lastName='Jin'
42  * 这里并不是改Mother.prototype,改动不同的层次,效果往往会有很大的差异。
43  * p1:'Jack',['Hangzhou','Guangzhou']; __proto__:{'jin',__proto__:18,['Shenzhen','Shanghai']}
44  * p2:'Mark';                          __proto__:{'jin',__proto__:18,['Shenzhen','Shanghai']}
45  */
46 
47 Person.prototype = { 
48     age: 28, 
49     address: { country: 'USA', city: 'Washington' }
50 };
51 var p3 = new Person('Obama'); 
52 /* 重写原型!因为在通过原型链实现继承时,不能使用对象字面量创建原型方法,因为这样会重写原型链。这个时候Person的原型已经完全变成一个新的对象了,
53  * 换成这样理解:var a=10; b=a; a=20; c=a。所以b不变,变得是c,所以p3跟着新的原型变化,与之前的原型无关。
54  * p1:'Jack',['Hangzhou','Guangzhou']; __proto__:{'jin',__proto__:18,['Shenzhen','Shanghai']}
55  * p2:'Mark';                          __proto__:{'jin',__proto__:18,['Shenzhen','Shanghai']}
56  * p3:'Obama';__proto__: 28 {country: 'USA', city: 'Washington'}
57  */
58 
59 
60 Mother.prototype.no = 9527;
61 /* 改写原型的原型,动态反应到实例中。
62  * 注意,这里我们改写的是Mother.prototype,p1p2会变,但上面p3跟之前的原型已经了无瓜葛了,所以不受影响。
63  * p1:'Jack',['Hangzhou','Guangzhou']; __proto__:{'jin',__proto__:18,['Shenzhen','Shanghai'],9527}
64  * p2:'Mark';                          __proto__:{'jin',__proto__:18,['Shenzhen','Shanghai'],9527}
65  * p3:'Obama';__proto__: 28 {country: 'USA', city: 'Washington'}
66  */
67 
68 Mother.prototype = { 
69     car: 2, 
70     hobby: ['run','walk']
71 };
72 var p4 = new Person('Tony');
73 /* 重写原型的原型!这个时候Mother的原型已经完全变成一个新的对象了!
74  * 由于上面Person与Mother已经断开联系了,这时候Mother怎么变已经不影响Person了。
75  * p4:'Tony';__proto__: 28 {country: 'USA', city: 'Washington'}
76  */
77  
78 Person.prototype = new Mother(); //再次绑定
79 var p5 = new Person('Luffy');
80 // 这个时候如果需要应用这些改动的话,那就要重新将Person的原型绑到mother上了
81 // p5:'Luffy';__proto__:{__proto__: 2, ['run','walk']}
82 
83 p1.__proto__.__proto__.__proto__.__proto__ //null,你说原型链的终点不是null?
84 Mother.__proto__.__proto__.__proto__    //null,你说原型链的终点不是null?

在第13行和第26行:p1.age = 20;  p1.home = ['Hangzhou', 'Guangzhou'];这两个是对实例p1添加了两个属性,对其原型没有任何影响,等同于为普通对象添加属性。

第20行:p1.home[0] = 'Shenzhen' 它不会在p1下创建一个home数组属性,然后将其首位设为 'Shenzhen',而是修改了Mother原型中home属性的值。因为home在p1下并未被定义,所以也不能直接一步定义home[0],如果要在p1下创建一个 home 数组,可以这样写:

p1.home = []; 
p1.home[0] = 'Shenzhen';

 而之所以 p1.home[0] = 'Shenzhen' 不直接报错,是因为在原型链中有一个搜索机制。当我们输入 p1.object 的时候,原型链的搜索机制是先在实例中搜索相应的值,找不到就在原型中找,还找不到就再往上一级原型中搜索……一直到了原型链的终点,就是到null还没找到的话,就返回一个 undefined。当我们输入 p1.home[0] 的时候,也是同样的搜索机制,先搜索 p1 看有没有名为 home 的属性和方法,然后逐级向上查找。最后我们在Mother的原型里面找到了,所以修改他就相当于修改了 Mother 的原型啊。

posted @ 2018-08-01 16:43  L_mj  阅读(255)  评论(0编辑  收藏  举报