前言
一直对原型链这一块,感到比较晦涩。理解起来比较困难不说,实际业务还很少用到。每次看完之后,没多久就忘了。针对这种特点的知识点,就得花时间梳理一下,做成在线笔记。需要用的时候,查看一下在线文档,现在我们进入正题。
JS原型链
js对象通过键名查找对象属性值的过程是:首先会从对象自身属性开始查找,如果查不到就会去原型对象__proto__
中去查找,如果原型对象也没有就会把当前的原型对象当作实例对象,继续通过__proto__
去当前原型对象的原型对象中去找,由此形成一条链条,直到在__proto__
链条上中找到或__proto__
的指向为null
时停止;要想搞清楚原型链,就得理清 __proto__
、prototype
和 constructor
三者的关系。
结合下面的例子,我们梳理一下 __proto__
、prototype
和 constructor
的关系。
js
// 构造函数
function Bar(){
this.name=name;
}
// 原型对象
Bar.prototype.say=function(){
console.log(this.name);
}
// 实例对象
const f=new Bar('bar1');
内置原型__proto__
每个实例对象(本例是f
)都有内置原型__proto__
,指向了创建该对象的构造函数原型(本例中是Bar.prototype
)。构造函数原型也有内置原型__proto__
(本例是Object.prototype
),这条链条的终点是null
。通过 __proto__
将对象和原型联系起来组成原型链,就可以扩展对象的属性。JS内置的构造函数Object
,Function
,Array
,String
,Number
,Boolean
,Date
,RegExp
,Error
的__proto__
都是Function.prototype
。而Function.prototype
指向Object.prototype
,如下图所示:
而Object.prototype
具有的方法和属性如下图所示:
根据原型链查找原理,JS内置的9大构造函数生成的实例对象,应该都有这些方法和属性。这里用Boolean
类型的实例对象验证一下JS内置的构造函数是不是具有Object.prototype
的toString
方法。如下图所示,确实有。
prototype
和constructor
的关系
上面我们看到,Object.prototype
打印输出了一个constructor
属性,这个属性指向哪里呢? 答案是Object
构造器。constructor
指向的都是构造函数。prototype
指向函数的原型对象,只有函数才有该属性。
__proto__
、prototype
和 constructor
关系
__proto__
、prototype
和 constructor
的关系,是上面两幅图的叠加,感觉prototype
和 constructor
的关系还是很好理清,主要是加入__proto__
关系之后,关系图看着有点不清晰了。所以重点是要理解__proto__
的指向关系。
小测验
做做下面这道题,检测一下学习效果:
js
function Man(){
this.age= 20;
}
function Male() {
this.age= 25;
}
Man.__proto__.print = function(){
console.log(this.age);
}
Man.print();
Male.print();
var man1 = new Man();
man1.print();
Man.print()
的执行过程是:Man
构造函数上并没有print
方法,所以去Man
的内置原型__proto__
去找,找到了print
方法定义:
js
Man.__proto__.print = function(){
console.log(this.age);
}
而Man.__proto__
上没有age
的定义,所以Man.print()
执行之后输出undefined
;
Male.print()
的执行过程是:Male
构造函数上并没有print
方法, 所以去Male
的内置原型__proto__
去找, 从文中的第一幅图可以看出 ,Male.__proto__
和Man.__proto__
都指向Function.prototype
, 也就是
js
Man.__proto__ === Function.prototype === Male.__proto__
Man.__proto__
定义了print
方法, 所以在Male.__proto__
上能找到print
方法, 而Male.__proto__
, Man.__proto
和Function.prototype
都没有定义age
属性, 所以Male.print()
执行之后输出undefined
;
man1.print()
的执行过程是:man1
构造函数原型对象Man.prototype
上并没有print
方法, 所以去Man.prototype
的内置原型__proto__
去找, Man.prototype.__proto__
指向Object.prototype
, Object.prototype
上也没有,继续沿着Object.prototype.__proto__
找, 找到了__proto__
内置原型链的末端null
,也没找到,所以执行报错:
js
Uncaught TypeError: man1.print is not a function
这个小测验中的三道题你都做对了吗?如果没有完全做对,再回头看看原文,尤其是内置原型那幅图。