原型、原型链、原型继承和类继承特点
# 基本属性
proto
- 是一个指向原型对象的指针
prototype
- 实例
constructor
- 指向创建它的构造函数
# 继承的方式
- 继承的本质就是原型链。
# 一. 构造函数继承
function Parent(name){
this.name=name
console.log('name', this.name);
}
const Child=new Parent('你好') //Parent.call(this)
// name 你好
1
2
3
4
5
6
2
3
4
5
6
原理:通过内部改变
this
的指向。缺点:无法继承父的原型,即父的原型增加一个方法,无法被子继承。
Parent.prototype.say = function () {};
1创建构造函数
new
的过程
//伪代码表示
var a = new Parent("Li","Cherry");
new Parent{
var obj = {};
obj.__proto__ = Parent.prototype;
var result = Parent.call(obj,"Li","Cherry");
return typeof result === 'object'? result : obj;
}
1
2
3
4
5
6
7
8
9
2
3
4
5
6
7
8
9
- 过程分析:
- 创建一个空对象
obj
; - 将新创建的空对象的隐式原型指向其构造函数的显示原型。
- 使用
call
改变this
的指向 - 如果无返回值或者返回一个非对象值,则将
obj
返回作为新对象;如果返回值是一个新对象的话那么直接直接返回该对象。
# 二. 原型链继承
function Parent(){
this.name='parent xx'
}
function Child(){
this.type='child xx'
}
Child.prototype=new Parent()
//console.log(new Child()) 打印结果
// Child{
// type:'child xx'
// __proto__:Parent{
// name:'parent xx'
// __proto__:Object
// }
// }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
2
3
4
5
6
7
8
9
10
11
12
13
14
15
原理:每个函数都有
prototype
属性,构造函数也有这个属性,这个属性是一个对象。把Parent
的实例赋值给了Child
的prototye
,实现继承new Child.__proto__ === new Parent()
的结果为true
优点:Child 可以继承 Parent 的原型
缺点:如果修改 child1实例的name属性,child2实例中的name属性也会跟着改变。
let child1=new Child() let child2=new Child() child1.name='xx' console.log('child1',child1.name) //child1 xx console.log('child2',child2.name) //child2 xx
1
2
3
4
5
# 组合:构造函数+原型链
function Parent(){
this.name='parent xx'
this.arr=[1,2,3]
}
function Child(){
Parent.call(this)
this.type='child type'
}
Child.prototype=new Parent()
let child1 = new Child();
let child2 = new Child();
child1.name = 'xx';
console.log('child1', child1.name); //child1 xx
console.log('child2', child2.name); //child2 parent xx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
2
3
4
5
6
7
8
9
10
11
12
13
14
- 优点:可以解决之前两种问题,既可以继承父类原型的内容,也不会造成原型里属性的修改。
- 缺点:执行了两次Parent构造
# ES6 class extend
主要理解ES5的方法
上次更新: 2024/04/15, 14:35:14