史上最为详细的javascript继承

前言

        为你们分享js中最多见最详细的继承方式,接下来将一下面的几个维度进行展现说明

文章有点长,请耐心阅读😁,有什么错误理解的地方但愿留言指出来bash

  • 产生缘由
  • 代码实现
  • 基本原理
  • 语言实现
  • 场景优势
  • 缺点

继承方式

  • 原型链继承
  • 借用构造函数模式继承
  • 组合继承
  • 原型式继承
  • 寄生式继承
  • 寄生组合

原型链继承

        相信小伙伴们都知道到原型链继承(ECMAScript 中描述了原型链的概念,并将原型链做为实现继承的主要方法),由于原型链继承很是的强大,可是也有它的缺点,接下来我们就按照上面的维度看看原型链继承究竟是什么鬼app

代码实现:(须要两个构造函数来完成一个原型链继承)

// SuperType 构造函数称为超类
 function SuperType (){
     this.name='super';
     this.friend=[];
     this.property = true; 
}
SuperType.prototype.getName=function(){
    return this.name;
}
SuperType.prototype.getSuperValue = function(){
 return this.property;
}; 
// SubType 构造函数称为子类
function SubType(name,age){
   this.name=name;
   this.age=age;
   this.subproperty = false; 
}
SubType.prototype=new SuperType();
SubType.prototype.constrcutor=SubType;
SubType.prototype.getAge=function(){
    return this.age;
}
SubType.prototype.getSubValue = function (){
 return this.subproperty;
}; 
var child = new SubType('shiny',12);
console.log(child.getName)//shiny
console.log(child.getAge())//12
复制代码

图解部分 属性

基本原理

使用相似做用域的原型链,进行继承查找

语言实现

        定义两个构造函数,分别为父类(SuperType)、子类(SubType),为了实现子类可以使用父类的属性(自己和原型上面的属性)。重写子类的原型,让子类的原型指向父类实例,这样子类的构造函数就是父类的实例地址,实现子类可使用父类的自己和原型上的属性

优势

        子类能够经过原型链的查找,实现父类的属性公用与子类的实例函数

缺点

        
  • 一些引用数据操做的时候会出问题,两个实例会公用继承实例的引用数据类
  • 谨慎定义方法,以避免定义方法也继承对象原型的方法重名
  • 没法直接给父级构造函数使用参数

借用构造函数模式继承

        虽然原型链继承很强大可是也有他的缺点,借用构造函数继承能够解决原型链继承的缺点,开线面的解释

代码实现:

// 把父类当中一个函数使用
function SuperType(name){
this.name=name
this.friend=['a','b']
}
SuperType.prototype.getFriend=function(){
  return this.firend
}
function SubType(name){
  // 执行父类函数
  SuperType.call(this,name);
}
var child = new SubType('shiny')
var childRed = new SubType('red')
console.log(child.name)//shiny
console.log(childRed.name)//red
child.firend.push('c')
console.log(child.friend)//a,b,c
console.log(childRed.friend)//a,b
console.log(childRed.getFriend)//undefined
复制代码

基本原理

使用call apply方法,经过执行方法修改tihs (上下文),是的父级的this变成子类实例的this,这样每一个实例都会获得父类的属性,实现引用属性备份

使用场景

父类中须要一些子类使用共享的引用类型,而且子类可能会操做父类共享的引用类型 可是父类的非this绑定的属性和方法是不可使用的(放在父类prototype的属性和方法)

语言实现

不要把父类当中构造函数,当中一个函数来处理这样更容易理解,在子类的构造函数中借用父类函数经过修改this来执行,这样子类的实例包含父类的属性

优势

  • 解决了原型链继承的 引用类型操做问题
  • 解决了父类传递参数问题

缺点

  • 仅仅使用借用构造函数模式继承,没法摆脱够着函数。方法在构造函数中定义复用不可谈
  • 对于超类的原型定义的方法对于子类是不可以使用的,子类的实例只是获得了父类的this绑定的属性 考虑到这些缺点,单独使用借用构造函数也是不多使用的

组合继承

        上面的两种继承方式(原型链继承+借用构造函数继承),都有本身优缺点,可是他们不是很完美,下面解释一下组合继承

代码实现:

代码实现:
function SuperType(name){
    this.name=name;
    this.firend=['a','b']
}
SuperType.prototype.getName=function(){
   return this.name
}
function SubType(name,age){
   this.age=age;
   SuperType.call(this,name)
}
SubType.prototype=new SuperType();
SubType.prototype.constrcutor = SubType;
SubType.prototype.getAge=function(){
   return this.age
}
var childShiny=new SubType('shiny',23);
var childRed = new SubType('red',22);
childShiny.firend.push('c');
childRed.firend.push('d');
console.log(childShiny.getName());
console.log(childShiny.getAge());
console.log(childRed.getName());
console.log(childRed.getAge());
console.log(childRed.friend);//[a,b,d]
console.log(childShiny.friend);//[a,b,c]
复制代码

基本原理

  • 使用原型链的继承实现,经过原型查找功能来知足原型链共享方法
  • 使用借用构造函数方法,使用实例备份父类共享引用类型备份

使用场景

获得原型链继承和构造函数继承的优势,是被开发人员承认的一种继承方式,可是也有他的缺点

语言实现

  • 定义两个构造函数,分别为父类(SuperType)、子类(SubType),为了实现子类可以使用父类的属性(自己和原型上面的属性)。重写子类的原型,让子类的原型指向父类实例,这样子类的构造函数就是父类的实例地址,实现子类可使用父类的自己和原型上的属性
  • 不要把父类当中构造函数,当中一个函数来处理这样更容易理解,在子类的构造函数中借用父类函数经过修改this来执行,这样子类的实例包含父类的属性

优势

  • 解决了原型链继承引用类型的实例操做致使引用改变
  • 解决了借构造函数继承方式的,父类原型子类实例可使用

缺点

* 父类的构造函数被实例换了两次 * 实例会有父类的构造函数的一些this属性、子类的构造函数(prototype)上也有一份实例的上有的属性

原型式继承

        话说上面的的组合继承不是已经被开发者承认了吗,原型式继承是啥?下面我们看看原型式继承是什么样的。

代码实现:

代码实现:
1  function object(o){
  function F(){};
  F.prototype=o;
   return new F()
}
var person = {
name: "Nicholas",
friends: ["Shelby", "Court", "Van"]
};
var personShiny = object(person);
var personRed = object(person);
console.log(personShiny.name)//Nicholas
console.log(personRed.name)//Nicholas
personShiny.friends.push('red');
personRed.friends.push('shiny');
console.log(personShiny.friends)//["Shelby", "Court", "Van","red","shiny"]
//ECMAScript 5 经过新增 Object.create()方法规范化了原型式继承。这个方法接收两个参数:一
//个用做新对象原型的对象和(可选的)一个为新对象定义额外属性的对象。在传入一个参数的状况下,
//Object.create()与 object()方法的行为相同。
2 
var person = {
name: "Nicholas",
friends: ["Shelby", "Court", "Van"]
};
var personShiny = Object.create(person);
var personRed = Object.create(person);
console.log(personShiny.name)//Nicholas
console.log(personRed.name)//Nicholas
personShiny.friends.push('red');
personRed.friends.push('shiny');
console.log(personShiny.friends)//["Shelby", "Court", "Van","red","shiny"]
复制代码

基本原理

经过Object.create()方法来建立一个有基础类的实例,这实例的__proto__指向基础类ui

使用场景

在不使用构造函数的状况下,只想让一个对象与另外一个对象保持相似的状况下

语言实现

须要建立一个基础对象,做为一个新对象的基础对象,经过object方法或者Object.create方法处理获得一个新实例,这个新实例上的__proto__指向基础对象

优势

再不用建立构造函数的状况下,实现了原型链继承,代码量减小一部分this

缺点

  • 一些引用数据操做的时候会出问题,两个实例会公用继承实例的引用数据类
  • 谨慎定义方法,以避免定义方法也继承对象原型的方法重名
  • 没法直接给父级构造函数使用参数

寄生继承

        我们看了上面的原型式继承,其实就是和原型链继承差异不大,只是省去了构造函数这一部,可是原型式继承也是有缺点的(不可以给备份的对象添加属性),下面寄生继承来解决。

代码实现:

代码实现:
// 和工厂模式很是相似,建立一个对象,加强一些功能并返回该对象
function createAnother(o){
   var clone = Object(o);
   clone.sayHi=function(){
     console.log('hi')
}
   return clone
}
var person = {
   name:'shiny',
   friends:['a','b']
}
var personShiny = createAnother(person);
console.log(personShiny.sayHi())//Ho
复制代码

基本原理

备份一个对象,而后给备份的对象进行属性添加,并返回spa

使用场景

在考不使用构造函数的状况下实现继承,前面示 范继承模式时使用的 object()函数不是必需的;任何可以返回新对象的函数都适用于此模式

语言实现

相似构造函数,经过一个执行方法,里面建立一个对象,为该对象添加属性和方法,而后返回

优势

  • 再不用建立构造函数的状况下,实现了原型链继承,代码量减小一部分
  • 能够给备份的对象添加一些属性

缺点

相似构造函数同样,建立寄生的方法须要在clone对象上面添加一些想要的属性,这些属性是放在clone上面的一些私有的属性prototype

寄生组合继承

        我们看了上面的组合继承看上去已经很完美了,可是也有缺点(父类被实例化两次、子类实例和子类的构造函数都有相同的属性),寄生组合就是来解决这些问题的

代码实现:

代码实现:
 function inheritPrototype({SubType,SuperType}){
      const prototype = Object(SuperType.prototype);
      prototype.constrcutor=SubType;
      SubType.prototype=prototype;
}
function SuperType(name){
    this.name=name;
    this.friends=['a','b']
}
SuperType.prototype.getName=function(){
   return this.name;
}
function SubType(name,age){
    this.age=age;
    SuperType.call(this,name)
}
inheritPrototype({SubType,SuperType});
SubType.prototype.getAge=function(){
   return this.age
}
var SubTypeShiny = new SubType('Shiny',23);
SubTypeShiny .friends.push('c')
var SubTypeRed = new SubType('Red',21);
SubTypeRed .friends.push('d')
console.log(SubTypeShiny.getName())//Shiny
console.log(SubTypeShiny.getAge())//22
console.log(SubTypeShiny.friends)//['a','b','c']
console.log( SubTypeRed.getName())//Red
console.log( SubTypeRed.getAge())//21
console.log( SubTypeRed.friends)//['a','b','d']
复制代码

基本原理

  • 子类构造函数内经过call、apply方法进行修改父类构造函数的this和执行父类构造函数,使的子类的实例拥有父类构造函数的一些属性,
  • 结合子类的原型修改为父类构造函数的原型,并把父类的原型的constructor指向子类构造函数

使用场景

在考不使用构造函数的状况下实现继承,前面示 范继承模式时使用的 object()函数不是必需的;任何可以返回新对象的函数都适用于此模式

语言实现

极度相似组合寄生方式,只是修改了子类原型链继承的方式,组合寄生是继承父类的实例,寄生组合寄生则是经过一子类的原型继承父类的原型,并把该原型的constructor指向子类构造函数

优势

  • 在少一次实例化父类的状况下,实现了原型链继承和借用构造函数
  • 减小了原型链查找的次数(子类直接继承超类的prototype,而不是父类的实例)

缺点

暂无code

下面是组合继承和寄生组合继承的原型图对比cdn

相关文章
相关标签/搜索