文章目录
- js继承的6种方式
- 1、原型链继承
- 特点:
- 缺点:
- 2、借用构造函数继承
- 特点:
- 缺点:
- 3、组合继承(组合原型链继承和借用构造函数继承)
- 特点:
- 缺点:
- 4、原型式继承
- 特点:
- 缺点:
- 5、寄生式继承
- 特点:
- 缺点:
- 6、寄生组合式继承(常用)
- 寄生:在函数内返回对象然后调用
- 组合:
js继承的6种方式
想要继承,就必须要提供个父类(继承谁,提供继承的属性)例子:
//父类
function Person(name) { //给构造函数添加了参数
this.name = name
this.sum = function() {
alert(this.name)
}
}
Person.prototype.age = 10; //给构造函数添加了原型属性
1、原型链继承
原型链继承其核心在于将父类的实例作为子类的原型
特点:
- 子类的实例也是父类的实例
- 可以方便的继承父类型的原型中的方法,但是属性的继承无意义
缺点:
- 只执行一次,无法给属性传值
- 属性的继承无意义
例子:
function Person(name, age, sex) {
this.name = name;
this.age = age;
this.sex = sex;
}
Person.prototype.sayHi = function() {
console.log('大家好,我是' + this.name);
};
//子类型
function Student() {
this.score = 100;
}
Student.prototype = new Person();
Student.prototype.constructor = Student;
var s1 = new Student()
s1.sayHi() //大家好,我是undefined
console.log(s1.name); //undefined
console.log(s1.age); //undefined
console.log(s1.constructor); //student
2、借用构造函数继承
借用构造函数继承其核心是在子类的内部调用父类,通过call改变父类中this的指向,等于是复制父类的实例属性给子类
特点:
- 创建子类实例时,可以向父类传递参数
- 可以实现多继承
- 可以方便的继承父类型的属性,但是无法继承原型中的方法
缺点:
- 实例并不是父类的实例,只是子类的实例
- 无法继承原型中的方法
- 无法实现函数复用,每个子类都有父类实例函数的副本,影响性能
例子:
//构造器
function Animation(name, age) {
this.name = name;
this.age = age;
}
Animation.prototype.eat = function() {
console.log(this.name);
}
function Cat(name, age, sex, score) {
Animation.call(this, name, age); //可以call多个父类对象
this.sex = sex;
this.score = score;
}
var c1 = new Cat('tom', 18, '男', 100); //可以传参
c1.eat(); //报错 not a function
console.log(c1 instanceof Animation); //false
console.log(c1 instanceof Cat);
3、组合继承(组合原型链继承和借用构造函数继承)
组合继承其核心是结合了两种模式的优点,传参和复用
特点:
- 可以继承父类原型上的属性,可以传参,可复用。
- 每个新实例引入的构造函数属性是私有的。
缺点:
- 调用了两次父类构造函数(耗内存),子类的构造函数会代替原型上的那个父类构造函数。
例子:
//组合原型链和构造函数继承
function SubType(name) {
Person.call(this, name); //借用构造函数模式
}
SubType.prototype = new Person(); //原型链继承
var sub = new SubType("gar");
console.log(sub.name); //"gar"继承了构造函数属性
console.log(sub.age); //继承了父类原型的属性
4、原型式继承
原型式继承其核心是用一个函数包装一个对象,然后返回这个函数的调用,这个函数就变成了个可以随意增添属性的实例或对象。object.create()就是这个原理。
特点:
- 类似于复制一个对象,用函数来包装。
缺点:
- 所有实例都会继承原型上的属性。
- 无法实现复用。(新实例属性都是后面添加的)
例子:
//先封装一个函数容器,用来输出对象和承载继承的原型
function conent(obj) {
function F() {}
F.prototype = obj; //继承了传入的参数
return new F(); //返回函数对象
}
var sup = new Person() //拿到父类的实例
var sup1 = conent(sup);
console.log(sup1.age); //继承了父类函数的属性
5、寄生式继承
寄生式继承其核心是就是给原型式继承外面套了个壳子。
特点:
- 没有创建自定义类型,因为只是套了个壳子返回对象(这个),这个函数顺理成章就成了创建的新对象。
缺点:
- 没用到原型,无法复用。
例子:
function content(obj) {
function F() {}
F.prototype = obj; //继承了传入的参数
return new F(); //返回函数对象
}
var sup = new Person();
//以上是原型式继承,给原型式继承套个壳子传递参数
function subobject(obj) {
var sub = content(obj);
sub.name = "gar";
return sub;
}
var sup2 = subobject(sup);
//这个函数经过声明之后就成了可增添属性的对象
console.log(typeof subobject); //function
console.log(typeof sup2); //object
console.log(sup2.name); //"gar",返回了个sub对象,继承了sub的属性
6、寄生组合式继承(常用)
寄生组合式继承其核心是修复了组合继承的问题
寄生:在函数内返回对象然后调用
组合:
- 函数的原型等于另一个实例。
- 在函数中用apply或者call引入另一个构造函数,可传参
例子:
//寄生
function content(obj) {
function F() {}
F.prototype = obj;
return new F();
}
//content就是F实例的另一种表示法
var con = content(Person.prototype);
//con实例( F实例)的原型继承了父类函数的原型
//上述更像是原型链继承,只不过只继承了原型属性
//组合
function Sub() {
Person.call(this); //这个继承了父类构造函数的属性
}
//解决了组合式俩次调用构造函数属性的缺点
//重点
Sub.prototype = con; //继承了con实例
con.constructor = Sub; //-定要修复实例
var sub1 = new Sub();
//Sub的实例就继承了构造函数属性,父类实例,con的函数属性
console.log(sub1.age); //10;