js中的几种继承方式

素颜马尾好姑娘i 2021-09-18 02:48 548阅读 0赞

定义父类:

  1. // 定义一个动物类
  2. function Animal (name) {
  3. // 属性
  4. this.name = name || 'Animal';
  5. // 实例方法
  6. this.sleep = function(){
  7. console.log(this.name + '正在睡觉!');
  8. }
  9. }
  10. // 原型方法
  11. Animal.prototype.eat = function(food) {
  12. console.log(this.name + '正在吃:' + food);
  13. };

1、原型链继承
核心: 将父类的实例作为子类的原型

  1. function Cat(){
  2. }
  3. Cat.prototype = new Animal();
  4. Cat.prototype.name = 'cat';
  5. // Test Code
  6. var cat = new Cat();
  7. console.log(cat.name);
  8. console.log(cat.eat('fish'));
  9. console.log(cat.sleep());
  10. console.log(cat instanceof Animal); //true
  11. console.log(cat instanceof Cat); //true

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

缺点:
1、要想为子类新增属性和方法,必须要在new Animal()这样的语句之后执行,不能放到构造器中

可以在Cat构造函数中,为Cat实例增加实例属性。如果要新增原型属性和方法,则必须放在new Animal()这样的语句之后执行

2、无法实现多继承
3、来自原型对象的所有属性被所有实例共享
4、创建子类实例时,无法向父类构造函数传参

推荐指数:★★(3、4两大致命缺陷)

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

  1. function Cat(name){
  2. Animal.call(this);
  3. this.name = name || 'Tom';
  4. }
  5. // Test Code
  6. var cat = new Cat();
  7. console.log(cat.name);
  8. console.log(cat.sleep());
  9. console.log(cat instanceof Animal); // false
  10. console.log(cat instanceof Cat); // true

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

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

推荐指数:★★(缺点3)

3、实例继承
核心:为父类实例添加新特性,作为子类实例返回

  1. function Cat(name){
  2. var instance = new Animal();
  3. instance.name = name || 'Tom';
  4. return instance;
  5. }
  6. // Test Code
  7. var cat = new Cat();
  8. console.log(cat.name);
  9. console.log(cat.sleep());
  10. console.log(cat instanceof Animal); // true
  11. console.log(cat instanceof Cat); // false

优点:
1、不限制调用方式,不管是new 子类()还是子类(),返回的对象具有相同的效果

缺点:
1、实例是父类的实例,不是子类的实例
2、不支持多继承

推荐指数:★★

4、拷贝继承

  1. function Cat(name){
  2. var animal = new Animal();
  3. for(var p in animal){
  4. Cat.prototype[p] = animal[p];
  5. }
  6. Cat.prototype.name = name || 'Tom';
  7. }
  8. // Test Code
  9. var cat = new Cat();
  10. console.log(cat.name);
  11. console.log(cat.sleep());
  12. console.log(cat instanceof Animal); // false
  13. console.log(cat instanceof Cat); // true

优点:
1、支持多继承

缺点:
1、效率较低,内存占用高(因为要拷贝父类的属性)
2、无法获取父类不可枚举的方法(不可枚举方法,不能使用for in 访问到)

推荐指数:★(缺点1)

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

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

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

缺点:
1、调用了两次父类构造函数,生成了两份实例(子类实例将子类原型上的那份屏蔽了)

推荐指数:★★★★(仅仅多消耗了一点内存)

6、寄生组合继承
核心:通过寄生方式,砍掉父类的实例属性,这样,在调用两次父类的构造的时候,就不会初始化两次实例方法/属性,避免的组合继承的缺点

  1. function Cat(name){
  2. Animal.call(this);
  3. this.name = name || 'Tom';
  4. }
  5. (function(){
  6. // 创建一个没有实例方法的类
  7. var Super = function(){};
  8. Super.prototype = Animal.prototype;
  9. //将实例作为子类的原型
  10. Cat.prototype = new Super();
  11. })();
  12. // Test Code
  13. var cat = new Cat();
  14. console.log(cat.name);
  15. console.log(cat.sleep());
  16. console.log(cat instanceof Animal); // true
  17. console.log(cat instanceof Cat); //true
  18. // 该实现没有修复constructor。
  19. Cat.prototype.constructor = Cat; // 需要修复下构造函数

优点:
1、堪称完美

缺点:
1、实现较为复杂

推荐指数:★★★★(实现复杂,扣掉一颗星)

示例

  1. function Animal (name) {
  2. // 属性
  3. this.name = name || 'Animal';
  4. // 实例方法
  5. this.sleep = function(){
  6. console.log(this.name + '正在睡觉!');
  7. }
  8. //实例引用属性
  9. this.features = [];
  10. }
  11. function Cat(name){
  12. }
  13. Cat.prototype = new Animal();
  14. var tom = new Cat('Tom');
  15. var kissy = new Cat('Kissy');
  16. console.log(tom.name); // "Animal"
  17. console.log(kissy.name); // "Animal"
  18. console.log(tom.features); // []
  19. console.log(kissy.features); // []
  20. tom.name = 'Tom-New Name';
  21. tom.features.push('eat');
  22. //针对父类实例值类型成员的更改,不影响
  23. console.log(tom.name); // "Tom-New Name"
  24. console.log(kissy.name); // "Animal"
  25. //针对父类实例引用类型成员的更改,会通过影响其他子类实例
  26. console.log(tom.features); // ['eat']
  27. console.log(kissy.features); // ['eat']
  28. 原因分析:
  29. 关键点:属性查找过程
  30. 执行tom.features.push,首先找tom对象的实例属性(找不到),
  31. 那么去原型对象中找,也就是Animal的实例。发现有,那么就直接在这个对象的
  32. features属性中插入值。
  33. console.log(kissy.features); 的时候。同上,kissy实例上没有,那么去原型上找。
  34. 刚好原型上有,就直接返回,但是注意,这个原型对象中features属性值已经变化了。

发表评论

表情:
评论列表 (有 0 条评论,548人围观)

还没有评论,来说两句吧...

相关阅读

    相关 JS继承方式

    1.原型继承 核心:将父类的实例作为子类的原型(并不是把父类中的属性和方法克隆一份一模一样的给子类,而是让子类父类之间增加了原型链接) 特点:父类中私有的和公有的都继

    相关 JS实现继承方式

    前言 JS作为面向对象的弱类型语言,继承也是其非常强大的特性之一。那么如何在JS中实现继承呢?让我们拭目以待。 JS继承的实现方式 既然要实现继承,那么首先我们得有一

    相关 JS实现继承方式

    前言 JS作为面向对象的弱类型语言,继承也是其非常强大的特性之一。那么如何在JS中实现继承呢?让我们拭目以待。 JS继承的实现方式 既然要实现继承,那么首先我们

    相关 JS实现继承方式

    前言 JS作为面向对象的弱类型语言,继承也是其非常强大的特性之一。那么如何在JS中实现继承呢?让我们拭目以待。 JS继承的实现方式 既然要实现继承,那么首先我们得有一

    相关 JS实现继承方式

    [点击去原文][Link 1] 前言 JS作为面向对象的弱类型语言,继承也是其非常强大的特性之一。那么如何在JS中实现继承呢?让我们拭目以待。 JS继承的实现方式

    相关 JS实现继承方式

    前言 JS作为面向对象的弱类型语言,继承也是其非常强大的特性之一。那么如何在JS中实现继承呢?让我们拭目以待。 既然要实现继承,那么首先我们得有一个父类,代码如下: