美文网首页
ES5 中的继承实现方式

ES5 中的继承实现方式

作者: 时间的溺水者 | 来源:发表于2022-03-28 20:19 被阅读0次
    new 关键字
    当我们new一个构造器,主要有三步:
    
     -  创建一个空对象,将它的引用赋给 this,继承函数的原型
     - 通过 this 将属性和方法添加至这个对象
     - 最后返回 this 指向的新对象,也就是实例
    
    // ES5构造函数
    let Parent = function (name, age) {
        //1.创建一个新对象,赋予this,这一步是隐性的,
        // let this = {};
        //2.给this指向的对象赋予构造属性
        this.name = name;
        this.age = age;
        //3.如果没有手动返回对象,则默认返回this指向的这个对象,也是隐性的
        // return this;
    };
    const child = new Parent();
    
    类式继承
    function SuperClass() {
      this.superValue = true;
    }
    SuperClass.prototype.getSuperValue = function() {
      return this.superValue;
    }
    
    function SubClass() {
      this.subValue = false;
    }
    SubClass.prototype = new SuperClass();
    
    SubClass.prototype.getSubValue = function() {
      return this.subValue;
    }
    
    var instance = new SubClass();
    
    console.log(instance instanceof SuperClass)//true
    console.log(instance instanceof SubClass)//true
    console.log(SubClass instanceof SuperClass)//false
    
    

    虽然实现起来清晰简洁,但是这种继承方式有两个缺点:

    1、由于子类通过其原型prototype对父类实例化,继承了父类,所以说父类中如果共有属性是引用类型,就会在子类中被所有的实例所共享,因此一个子类的实例更改子类原型从父类构造函数中继承的共有属性就会直接影响到其他的子类
    2、由于子类实现的继承是靠其原型prototype对父类进行实例化实现的,因此在创建父类的时候,是无法向父类传递参数的。因而在实例化父类的时候也无法对父类构造函数内的属性进行初始化
    
    构造函数继承
    function SuperClass(id) {
      this.books = ['js','css'];
      this.id = id;
    }
    SuperClass.prototype.showBooks = function() {
      console.log(this.books);
    }
    function SubClass(id) {
      //继承父类
      SuperClass.call(this,id);
    }
    //创建第一个子类实例
    var instance1 = new SubClass(10);
    //创建第二个子类实例
    var instance2 = new SubClass(11);
    
    instance1.books.push('html');
    console.log(instance1)
    console.log(instance2)
    instance1.showBooks();//TypeError
    

    SuperClass.call(this,id)当然就是构造函数继承的核心语句了.由于父类中给this绑定属性,因此子类自然也就继承父类的共有属性。由于这种类型的继承没有涉及到原型prototype,所以父类的原型方法自然不会被子类继承,而如果想被子类继承,就必须放到构造函数中,这样创建出来的每一个实例都会单独的拥有一份而不能共用,这样就违背了代码复用的原则,所以综合上述两种,组合式继承方法诞生了。

    组合式继承
    
    function SuperClass(name) {
      this.name = name; 
      this.books = ['Js','CSS'];
    }
    SuperClass.prototype.getBooks = function() {
        console.log(this.books);
    }
    function SubClass(name,time) {
      SuperClass.call(this,name);
      this.time = time;
    }
    SubClass.prototype = new SuperClass();
    
    SubClass.prototype.getTime = function() {
      console.log(this.time);
    }
    
    

    如上,解决了之前说到的一些问题,但是SuperClass的构造函数执行了两遍就感觉非常的不妥.

    原型式继承
    function inheritObject(o) {
        //声明一个过渡对象
      function F() { }
      //过渡对象的原型继承父对象
      F.prototype = o;
      //返回过渡对象的实例,该对象的原型继承了父对象
      return new F();
    }
    
    

    原型式继承大致的实现方式如上,是不是想到了我们new关键字模拟的实现?

    其实这种方式和类式继承非常的相似,他只是对类式继承的一个封装,其中的过渡对象就相当于类式继承的子类,只不过在原型继承中作为一个普通的过渡对象存在,目的是为了创建要返回的新的实例对象。

    var book = {
        name:'js book',
        likeBook:['css Book','html book']
    }
    var newBook = inheritObject(book);
    newBook.name = 'ajax book';
    newBook.likeBook.push('react book');
    var otherBook = inheritObject(book);
    otherBook.name = 'canvas book';
    otherBook.likeBook.push('node book');
    console.log(newBook,otherBook);
    
    

    如上代码我们可以看出,原型式继承和类式继承一个样子,对于引用类型的变量,还是存在子类实例共享的情况。

    寄生式继承
    var book = {
        name:'js book',
        likeBook:['html book','css book']
    }
    function createBook(obj) {
        //通过原型方式创建新的对象
      var o = new inheritObject(obj);
      // 拓展新对象
      o.getName = function(name) {
        console.log(name)
      }
      // 返回拓展后的新对象
      return o;
    }
    

    其实寄生式继承就是对原型继承的拓展,一个二次封装的过程,这样新创建的对象不仅仅有父类的属性和方法,还新增了别的属性和方法。

    寄生组合式继承

    回到之前的组合式继承,那时候我们将类式继承和构造函数继承组合使用,但是存在的问题就是子类不是父类的实例,而子类的原型是父类的实例,所以才有了寄生组合式继承
    而寄生组合式继承是寄生式继承和构造函数继承的组合。但是这里寄生式继承有些特殊,这里他处理不是对象,而是类的原型。

    function inheritObject(o) {
      //声明一个过渡对象
      function F() { }
      //过渡对象的原型继承父对象
      F.prototype = o;
      //返回过渡对象的实例,该对象的原型继承了父对象
      return new F();
    }
    
    function inheritPrototype(subClass,superClass) {
        // 复制一份父类的原型副本到变量中
      var p = inheritObject(superClass.prototype);
      // 修正因为重写子类的原型导致子类的constructor属性被修改
      p.constructor = subClass;
      // 设置子类原型
      subClass.prototype = p;
    }
    

    组合式继承中,通过构造函数继承的属性和方法都是没有问题的,所以这里我们主要探究通过寄生式继承重新继承父类的原型。

    我们需要继承的仅仅是父类的原型,不用去调用父类的构造函数。换句话说,在构造函数继承中,我们已经调用了父类的构造函数。因此我们需要的就是父类的原型对象的一个副本,而这个副本我们可以通过原型继承拿到,但是这么直接赋值给子类会有问题,因为对父类原型对象复制得到的复制对象p中的constructor属性指向的不是subClass子类对象,因此在寄生式继承中要对复制对象p做一次增强,修复起constructor属性指向性不正确的问题,最后将得到的复制对象p赋值给子类原型,这样子类的原型就继承了父类的原型并且没有执行父类的构造函数。

    function SuperClass(name) {
      this.name = name;
      this.books=['js book','css book'];
    }
    SuperClass.prototype.getName = function() {
      console.log(this.name);
    }
    function SubClass(name,time) {
      SuperClass.call(this,name);
      this.time = time;
    }
    inheritPrototype(SubClass,SuperClass);
    SubClass.prototype.getTime = function() {
      console.log(this.time);
    }
    var instance1 = new SubClass('React','2017/11/11')
    var instance2 = new SubClass('Js','2018/22/33');
    
    instance1.books.push('test book');
    
    console.log(instance1.books,instance2.books);
    instance2.getName();
    instance2.getTime();
    
    image.png

    这种方式继承其实如上图所示,其中最大的改变就是子类原型中的处理,被赋予父类原型中的一个引用,这是一个对象,因此有一点你需要注意,就是子类在想添加原型方法必须通过prototype.来添加,否则直接赋予对象就会覆盖从父类原型继承的对象了.

    相关文章

      网友评论

          本文标题:ES5 中的继承实现方式

          本文链接:https://www.haomeiwen.com/subject/aejggktx.html