JS重点整理之JS原型链彻底搞清楚

时间:2024-10-21 09:12:44


对象

要清楚原型链,首先要弄清楚对象:

  • 普通对象

    • 最普通的对象:有__proto__属性(指向其原型链),没有prototype属性。
    • 原型对象( 原型对象还有constructor属性(指向构造函数对象))

  • 函数对象:
    • 凡是通过new Function()创建的都是函数对象。
                          拥有__proto__、prototype属性( 指向原型对象)。

                          FunctionObjectArrayDateString、自定义函数

                          特例: ( 是原型对象,却是函数对象,下面会有解释 )
函数对象
function f1(){};
var f2 = function(){};
var f3 = function("n1","n2","return n1+n2");

(typeof f1);  //function
(typeof f2);  //function
(typeof f3);   //function
(typeof Object);   //function
(typeof Array);   //function
(typeof String);   //function
(typeof Date);   //function
(typeof Function);   //function

Array是函数对象,是Function的实例对象,Array是通过newFunction创建出来的。因为Array是Function的实例,所以Array.__proto__ ===  


普通对象
var o1 = new f1(); 
var o2 = {};       
var o3 = new Object(); 

(typeof o1);  //Object
(typeof o2);   //Object
(typeof o3);   //Object

原型对象


        每创建一个函数都会有一个prototype属性,这个属性是一个指针,指向一个对象(通过该构造函数创建实例对象的原型对象)。原型对象是包含特定类型的所有实例共享的属性和方法。原型对象的好处是,可以让所有实例对象共享它所包含的属性和方法。
        第一块中有提到,原型对象属于普通对象。是个例外,它是原型对象,却又是函数对象,作为一个函数对象,它又没有prototype属性。
function person(){};

(typeof ) //Object
(typeof ) // Object
(typeof ) // 特殊 Function
(typeof ) //undefined 函数对象却没有prototype属性

解释:

     functionperson(){};

        其实原型对象就是构造函数的一个实例对象。就是person的一个实例对象。相当于在person创建的时候,自动创建了一个它的实例,并且把这个实例赋值给了prototype。

 function person(){};
var temp = new person();
 = temp;

function Function(){};
var temp = new Function();
 = temp; //由new Function()产生的对象都是函数对象

        从一张图看懂原型对象、构造函数、实例对象之间的关系

function Dog(){};

 = "小黄";
 =  13;
 = function(){
	return ;
}

var dog1 = new Dog();
var dog2 = new Dog();

 = "小黑";
(); // 小黄 来自原型
(); // 小黑 来自实例



//图中的一些关系
dog1.__proto__ === 

.__proto__ ===  //继承Object 下面原型链说

dog1.__proto__.__proto__ === 

 === Dog 

(dog1)

获取对象的原型
dog1.__proto__  //不推荐
(dog1) ===    //推荐


原型链

原型链是实现继承的主要方法。

 

先说一下继承,许多OO语言都支持两张继承方式:接口继承、实现继承。

|- 接口继承:只继承方法签名

|- 实现继承:继承实际的方法

由于函数没有签名,在ECMAScript中无法实现接口继承,只支持实现继承,而实现继承主要是依靠原型链来实现。


原型链基本思路:

利用原型让一个引用类型继承另一个引用类型的属性和方法。

每个构造函数都有一个原型对象,原型对象都包含一个指向构造函数想指针(constructor),而实例对象都包含一个指向原型对象的内部指针(__proto__)。如果让原型对象等于另一个类型的实例,此时的原型对象将包含一个指向另一个原型的指针(__proto__),另一个原型也包含着一个指向另一个构造函数的指针(constructor)。假如另一个原型又是另一个类型的实例……这就构成了实例与原型的链条。


原型链基本思路(图解):



举例说明:

function animal(){
	 = "animal";
}
 = function(){
	return ;
}

function dog(){
	 = "dog";
}
 = new animal();

 = function(){
	return ;
}

var xiaohuang = new dog();

//原型链关系
xiaohuang.__proto__ === 
.__proto__ === 
.__proto__ === 
.__proto__ === null
 
   

图解:


详细图


从xiaohuang这个实例,看出整个链条


总结:

Xiaohuang这个Dog的实例对象继承了Animal,Animal继承了Object。