💎一站式轻松地调用各大LLM模型接口,支持GPT4、智谱、星火、月之暗面及文生图 广告
# 类的继承 继承是面向对象的三大特性之一,也是实现软件复用的重要手段。 Java中是单继承,每个子类只有一个直接父类。有的时候把父类也成为基类或者超类。 ## 继承的特点 > 父类和子类是一种一般和特殊的关系。例如水果和苹果,苹果是一种特殊的水果。 在Java中定义子类继承父类的语法为 ~~~ [修饰符] class SubClass extends SuperClass { // 类定义部分 } ~~~ extends字面解释为扩展,其实就是子类除了拥有父类所有的成员变量和方法之外,还具备自己的特殊属性。 > 在子类中是不能访问父类中定义的private相关的成员变量和方法的,这点也完全符合private的使用方式。 ## 重写父类的方法 在子类中包含与父类同方法签名的现象称为「方法重写(Override)」,有时也称为方法覆盖。我们可以说子类重写了父类的方法,也可以说子类覆盖了父类的方法。 > 方法签名:包含方法名、形参列表。 子类重写方法的意义:子类继承了父类之后,调用父类的方法,就不需要重复写,节省代码量。 如果使用条件变更,父类的方法不能满足使用需求,那么就需要重写方法了,把父类的方法覆盖掉。重写,也能实现代码的多样性。就好比父亲和儿子的关系,儿子会继承父亲的所有特性,但是儿子也可以自己发展。 > 重写父类方法需要遵循的规则: > 返回值和访问控制修饰符可以不一样,但是要符合“两同两小一大原则” > 作为两同就是方法名和方法的参数列表相同 > 两小:子类的返回值类型和抛出的异常类要比父类的返回值类型和抛出的异常类更小或者相等。这里说的返回值类型小值指的是子类和父类的关系。 > 一大:子类的访问权限要比父类的访问权限要大或者相等 ## super限定 super的使用和this的使用很相似。 如果需要在子类中调用父类被覆盖的方法,可以使用super关键字。 同样也可以通过super.fun()调用没有被覆盖的方法,但是这样写没有什么特别的意义,因为本身在子类中就可以直接调用,无需使用super关键字。 ## 关于继承的几点说明 子类能够继承父类的信息包括: * 没有使用private修饰符修饰的成员变量和方法,子类都能够继承 * 子类重写了父类的方法,则父类的方法会被隐藏,隐藏的方法或者成员变量可以通过super关键字进行访问 * 引入super关键字的原因是可以访问被隐藏的成员变量和属性,而且super只能在子类的方法定义中使用。 ## 调用父类的构造器 > 在Java的世界里,所有的类都是Object类的子类。 在子类的构造器中,无论如何在第一行语句中都会调用一次父类的构造器。 如果没有显式的调用,那么会调用父类的没有参数的构造器,如果父类没有定义空参数的构造器,则此时会无法编译。 对象创建的过程就是调用对象类的构造器的过程,一个构造器最基本的组成 ~~~ [修饰符] 类名 { super(...); 调用父类的构造器,如果没有会调用父类的无参构造器 } ~~~ super()必须出现在第一行,而且只能有一个。 this和super在同一构造器中只可能出现一次。 子类构造器调用父类构造器可能出现的情况有 * 子类构造器执行体的第一行显式的使用super调用父类的构造器,此时系统将根据super(params)调用对应的构造器。 * 子类构造器执行提的第一行显式的使用this调用重载的构造器,系统将根据this(params)调用对应的重载构造器,本类中的对应的构造器再去调用父类的构造器。 * 子类构造器中既没有super又没有this,子类在执行构造器语句前执行父类的无参构造器。 * 无论如何子类都会调用一次父类的构造器。 > 在实际的开发过程中,我们一般都会保证在一个类中有一个无参的构造器的存在。因为如果在其类的子类中没有定义构造器,或者定义了构造器没有显式的去调用父类的构造器,子类也能被成功的编译。