继承的简介及注意事项

继承

概述

  • 多个类中存在相同属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需再定义这些属

性和行为,只要继承那个类即可。

  • 通过extends关键字可以实现类与类的继承

    • class 子类名 extends 父类名 {}
  • 单独的这个类称为父类,基类或者超类;这多个类可以称为子类或者派生类。

  • 有了继承以后,我们定义一个类的时候,可以在一个已经存在的类的基础上,还可以定义自己的新

​ 成员。

继承的好处

  • 提高了代码的复用性

    • 多个类相同的成员可以放到同一个类中
  • 提高了代码的维护性

    • 如果功能的代码需要修改,修改一处即可
  • 让类与类之间产生了关系,是多态的前提

    • 其实这也是继承的一个弊端:类的耦合性很强

特点

  • Java只支持单继承,不支持多继承。

    • 一个类只能有一个父类,不可以有多个父类。
    • class SubDemo extends Demo{} //ok
    • class SubDemo extends Demo1,Demo2…//error
  • Java支持多层继承(继承体系)

    • class A{}
    • class B extends A{}
    • class C extends B{}

注意事项

  • 子类只能继承父类所有非私有的成员(成员方法和成员变量)

    • 其实这也体现了继承的另一个弊端:打破了封装性
  • 子类不能继承父类的构造方法,但是可以通过super(后面讲)关键字去访问父类构造方法。

    • 不能通过小范围=new 大范围初始化对象
  • 不要为了部分功能而去继承

  • 我们到底在什么时候使用继承呢?

    • 继承中类之间体现的是:”is a”的关系。

成员之间的变量关系

  • 在子类方法中访问一个变量
    • 首先在子类局部范围找
    • 然后在子类成员范围找
    • 最后在父类成员范围找(肯定不能访问到父类局部范围)
    • 如果还是没有就报错。(不考虑父亲的父亲…)

super关键字

  • super的用法和this很像

    • this代表本类对应的引用。
    • super代表父类存储空间的标识(可以理解为父类引用)
  • 用法(this和super均可如下使用)

    • 访问成员变量
    • this.成员变量 super.成员变量
  • 访问构造方法(子父类的构造方法问题讲)

    • this(…) super(…)
  • 访问成员方法(子父类的成员方法问题讲)

    • this.成员方法() super.成员方法()

构造方法关系

  • 子类中所有的构造方法默认都会访问父类中空参数的构造方法

    • 因为子类会继承父类中的数据,可能还会使用父类的数据。所以,子类初始化之前,一定要

      先完成父类数据的初始化。

    • 每一个构造方法的第一条语句默认都是:super()。

成员方法关系

  • 通过子类对象去访问一个方法
    • 首先在子类中找
    • 然后在父类中找
    • 如果还是没有就报错。(不考虑父亲的父亲…)

重写

  • 方法重写概述
    • 子类中出现了和父类中一模一样的方法声明,也被称为方法覆盖,方法复写。
    • 使用特点:
      • 如果方法名不同,就调用对应的方法
      • 如果方法名相同,最终使用的是子类自己的
    • 方法重写的应用:
      • 当子类需要父类的功能,而功能主体子类有自己特有内容时,可以重写父类中的方法,这样,即沿袭了父类的功能,又定义了子类特有的内容。

方法的重写要遵循如下规则:

1、 方法名相同、形参列表相同

2、 子类方法返回值类型应比父类方法返回值类型相等或者更小。返回值类型比父类的要小,这

个返回值一定是有父子关系才行的,基本类型之间没有父子关系。

3、 子类方法声明抛出的异常类应该比父类方法更小或相等

4、 子类权限比父类权限大或者相等

当子类覆盖了父类方法后,子类对象将无法访问父类中被覆盖的方法,但可以在子类方法中调用

父类中被覆盖的方法,需要使用super关键字。

如果父类方法具有private访问权限,则该方法对其子类是隐藏的,也就无法重写该方法。如果子

类中定义了一个与父类private方法相同的方法名、相同形参列表、相同返回值类型的方法,依然

不是重写,只是在子类中定义了一个新的方法而已。

final

final关键字是最终的意思,可以修饰类,成员变量,成员方法。

  • 修饰类,类不能被继承
  • 修饰变量,变量变为常量,只能被赋值一次
  • 修饰方法,方法不能被重写

所有类都默认继承Object类

### 继承方式的选择与适用场景 在 JavaScript 中实现继承时,需要根据具体需求选择合适的继承方式。每种继承模式都有其特点和适用范围,例如原型链继承、构造函数继承、组合继承、寄生组合继承等。不恰当的继承方式可能导致代码冗余或性能问题,因此在设计类结构时应充分考虑继承关系的合理性[^1]。 ### 避免原型污染 使用原型链继承时,如果直接修改对象的 `prototype` 属性,可能会导致不可预料的问题。例如,多个子类共享同一个父类的实例作为原型时,其中一个子类对原型状态的修改会影响其他子类。这种现象称为“原型污染”,应当通过深拷贝或封装等方式加以规避。 ```javascript function Parent() { this.colors = ['red', 'blue']; } function Child() {} Child.prototype = new Parent(); // 原型链继承 const child1 = new Child(); child1.colors.push('green'); const child2 = new Child(); console.log(child2.colors); // 输出 ["red", "blue", "green"],说明原型被污染 ``` ### 构造函数继承的局限性 构造函数继承(又称经典继承)通过在子类构造函数中调用父类构造函数来实现继承,通常使用 `call()` 或 `apply()` 方法传递上下文。这种方式可以避免原型污染问题,并允许向父类构造函数传递参数。然而,它无法继承父类原型上的方法和属性,导致子类实例无法访问这些共享资源。 ```javascript function Parent(name) { this.name = name; } Parent.prototype.sayHello = function() { console.log(`Hello, ${this.name}`); }; function Child(name, age) { Parent.call(this, name); // 构造函数继承 this.age = age; } const child = new Child('Alice', 5); child.sayHello(); // 报错:child.sayHello is not a function ``` ### 组合继承的优势与缺点 组合继承结合了原型链继承和构造函数继承的优点,既能继承父类原型上的方法,又能保证每个子类实例拥有独立的状态。尽管如此,这种方法仍然存在一些缺陷,比如父类构造函数会被调用两次——一次是在创建子类原型时,另一次是在子类构造函数内部。这可能影响性能并造成不必要的初始化操作。 ```javascript function Parent(name) { this.name = name; this.colors = ['red', 'blue']; } Parent.prototype.sayHello = function() { console.log(`Hello, ${this.name}`); }; function Child(name, age) { Parent.call(this, name); // 第二次调用 Parent() this.age = age; } Child.prototype = new Parent(); // 第一次调用 Parent() Child.prototype.constructor = Child; const child1 = new Child('Alice', 5); child1.colors.push('green'); const child2 = new Child('Bob', 3); console.log(child2.colors); // 输出 ["red", "blue"] ``` ### 寄生组合继承的优化方案 寄生组合继承是一种更高效的继承模式,它通过借用构造函数来继承属性,而通过原型链的折中方式继承方法。这种方式只调用一次父类构造函数,并且能够保持原型链的完整性,从而提高程序执行效率。推荐在实际开发中采用此方法以减少不必要的开销。 ```javascript function inheritPrototype(subType, superType) { const prototype = Object.create(superType.prototype); // 创建父类原型的副本 prototype.constructor = subType; // 增强对象,修正构造函数指向 subType.prototype = prototype; // 指定给子类的原型 } function Parent(name) { this.name = name; this.colors = ['red', 'blue']; } Parent.prototype.sayHello = function() { console.log(`Hello, ${this.name}`); }; function Child(name, age) { Parent.call(this, name); this.age = age; } inheritPrototype(Child, Parent); // 实现寄生组合继承 const child = new Child('Alice', 5); child.sayHello(); // 输出 "Hello, Alice" ``` ### 相关问题
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值