zoukankan      html  css  js  c++  java
  • JS实现继承的方式

    首先要有一个父类

    // 定义一个动物类
    function Animal (name) {
      // 属性
      this.name = name || 'Animal';
      // 实例方法
      this.sleep = function(){
        console.log(this.name + '正在睡觉!');
      }
    }
    // 原型方法
    Animal.prototype.eat = function(food) {
      console.log(this.name + '正在吃:' + food);
    };

    1.原型链继承

       核心:将父类的实例作为子类的原型

    思路:借助原型可以基于已有的对象创建对象,同时还不必因此创建自定义类型。 object()函数内部,先创建一个临时的构造函数,然后将传入的对象作为这个构造函 数的原型,最后返回了这个临时类型的一个新实例。

    //原型链继承
            function Cat(){
    
            }
            Cat.prototype = new Animal();
            Cat.prototype.name = 'cat';
    
            var cat = new Cat();
            console.log(cat.name);//cat
            console.log(cat.eat('fish'));//undefined
            console.log(cat.sleep());//undefined
            console.log(cat instanceof Animal);//true
            console.log(cat instanceof Cat);//true

    特点:

    1. 非常纯粹的继承关系,实例是子类的实例,也是父类的实例
    2. 父类新增原型方法/原型属性,子类都能访问到
    3. 简单,易于实现

    缺点:

    1. 要想为子类新增属性和方法,可以在Cat构造函数中,为Cat实例增加实例属性。如果要新增原型属性和方法,则必须放在new Animal()这样的语句之后执行。必须要在new Animal()这样的语句之后执行,不能放到构造器中
    2. 无法实现多继承
    3. 来自原型对象的引用属性是所有实例共享的(详细请看附录代码: 示例1
    4. 创建子类实例时,无法向父类构造函数传参

      2、构造继承

      核心:使用父类的构造函数来增强子类实例,等于是复制父类的实例属性给子类(没用到原型)

    5. 思路:在子类构造函数的内部调用超类型构造函数。可以通过使用 apply()call()方法 在新创建的对象上执行构造函数。 缺点:方法都在构造函数中定义,函数的复用就无从谈起。在超类型的原型中定义的方 法,对子类而言也是不可见的,结果所有的类型都只能使用构造函数模式。

    6. function Cat(name){
                  Animal.call(this);
                  this.name = name || 'Tom';
              }
              var cat = new Cat();
              console.log(cat.name);//Tom
              console.log(cat.sleep());//undefined
              console.log(cat instanceof Animal); // false
              console.log(cat instanceof Cat); // true

      特点:

      1. 解决了1中,子类实例共享父类引用属性的问题
      2. 创建子类实例时,可以向父类传递参数
      3. 可以实现多继承(call多个父类对象)

      缺点:

      1. 实例并不是父类的实例,只是子类的实例
      2. 只能继承父类的实例属性和方法,不能继承原型属性/方法
      3. 无法实现函数复用,每个子类都有父类实例函数的副本,影响性能

        3、组合继承

      4. 思路:指的是将原型链和借用构造函数的技术组合到一起, 从而发挥二者之长。 思路:使用原型链实现对原型属性属性和方法的继承,通过借用构造函数来实现实例属 性的继承。 优点:既通过在原型上定义方法实现了函数复用,又能保证每一个实例都有它自己的数 组。 组合继承避免了原型链和借用构造函数的缺陷,融合了他们的优点,成为 JavaScript 中常用的继承模式。

      5. 核心:通过调用父类构造,继承父类的属性并保留传参的优点,然后通过将父类实例作为子类原型,实现函数复用

      6. function Cat(name){
          Animal.call(this);
          this.name = name || 'Tom';
        }
        Cat.prototype = new Animal();
        
        // 组合继承也是需要修复构造函数指向的。
        
        Cat.prototype.constructor = Cat;
        
        // Test Code
        var cat = new Cat();
        console.log(cat.name);
        console.log(cat.sleep());
        console.log(cat instanceof Animal); // true
        console.log(cat instanceof Cat); // true

        特点:

        1. 弥补了方式2的缺陷,可以继承实例属性/方法,也可以继承原型属性/方法
        2. 既是子类的实例,也是父类的实例
        3. 不存在引用属性共享问题
        4. 可传参
        5. 函数可复用

        缺点:

        1. 调用了两次父类构造函数,生成了两份实例(子类实例将子类原型上的那份屏蔽了)
  • 相关阅读:
    Kivy Minimum required opengl version (2.0) not found
    Rust 在Jebrains 家族工具里报 No Cargo projects found!
    Rust Cargo 下载慢的解决办法
    Python面试题库学习1-5
    Kivy Pyinstaller打包报: No module named pkg_resources.py2_warn
    Kettle 部署和远程调用(spoon 8.3)
    spring boot 发布的jar 二 发布到 windows
    spring boot 发布的jar 一 发布到 linux centos7
    springBoot项目打jar包后,修改配置文件中的配置项
    mysql sqldump 备份
  • 原文地址:https://www.cnblogs.com/zhaosijia----1234/p/8858099.html
Copyright © 2011-2022 走看看