了解原型链之前,先来了解一下什么是原型
原型
原型主要分为显示原型和隐式原型
- 显示原型:prototype
- 函数天生就拥有的一个属性
- 我们可以将一些属性和方法挂载在原型上;在创建实例之后,实例就可以调用这些属性和方法
- 我们也可以将一些公用的属性和方法添加在原型上,减少构造函数在执行时的性能开销
- 实例对象不能修改原型上的属性值
function Car(color){
this.name='su7'
this.lang='4800'
this.height=1400
this.color=color
}
const car=new Car('blue')
console.log(car)//Car { name: 'su7', lang: '4800', height: 1400, color: 'blue' }
可以看到,v8在执行这段函数的时候,把第2、3、4行这些默认的东西都给执行了,如果我们创建car1,car2,car3...的时候,每次都要重复执行这几行代码,这会大大增加v8的运行效率,所以我们要用到原型,将Car作为car的原型
Car.prototype.name='su7'
Car.prototype.lang=4800
Car.prototype.height=1400
function Car(color){
this.color=color
}
const car=new Car('blue')
car.name='yu7'
console.log(car.name)//yu7
console.log(Car.prototype)//{ name: 'su7', lang: 4800, height: 1400 }
console.log(car)//Car { color: 'blue', name: 'yu7' }
可以看到,把Car作为原型之后,可以大大省去v8的执行时间,就算创建再多的car都不用次次都执行名字长度高度,而且就算把对象中的名字改掉,原型当中的名字也不会改变
- 实例对象中显示拥有的属性 来自于 构造函数中定义的属性
- 实例对象中隐式拥有的属性 来自于 构造函数的原型上
- 隐式原型:__proto__
- 每个对象都拥有__proto__属性,该属性也是一个对象
- v8 在访问对象中的一个属性时,会先访问对象中显示存在的属性,如果没有,就会去对象的隐式原型上查找
- 实例对象的隐式原型 === 构造函数的显示原型
- constructor 构造器属性,记录该实例对象是由谁创建的
Person.prototype.say=function(){
console.log('123')//②在Person.prototype中找到了say()
}
function Person(){
this.name='强哥'
}
const p=new Person()//p.__proto__=Person.prototype
p.say()//123 ①发现p中找不到say,就去p的隐式原型中找 ③Persone.prototype有say(),就可以输出123了
这①②③的步骤也就是接下来要讲的原型链了
原型链
简单来说:每一个对象都有自己的隐式原型,如果在隐性原型中找不到目标,就会去它的隐性原型中的显性原型中寻找(隐性原型也是一个对象),一路向上直到找到null
Grand.prototype.house=function(){
this.house='汤臣一品'
}
function Grand(){
this.card=1000000
}
Father.prototype=new Grand
function Father() {
this.lastName = '张'
}
Child.prototype=new Father
function Child() {
this.age = 18
}
const p = new Child()
p.house()
console.log(p.house)//汤臣一品
console.log(p.car)//undefined此时执行的是p.__proto__=Child.prototype=Child.__proto__=Object.prototype=null
原型链主要分为以下几种情况
function Foo()
let f1 = new Foo();
f1.__proto__ = Foo.prototype;
Foo.prototype.__proto__ = Object.prototype; //Foo.prototype本质也是普通对象
Object.prototype.__proto__ = null; // 原型链到此停止
————————————————————我是分割线———————————————————
Foo.__proto__=Function.prototype
Function.prototype.__proto__=Object.prototype
————————————————————我是分割线———————————————————
Function.__proto__=Function.prototype
最后供上一张经典图方便理解