0022 Java学习笔记-面向对象-继承、多态、组合

时间:2021-04-30 23:13:55

继承的特点

  • 单继承:每个子类最多只有一个直接父类,注意是直接父类,间接父类个数不限
  • 注意父类的概念:A-->B-->C-->D,在这里,ABC都是D的父类,C是D的直接父类,AB是D的间接父类
  • 父类和子类是一般和特殊的关系;子类扩展了父类,子类是一种特殊的父类
  • Object是所有类的直接或间接父类;定义一个类时,若没有直接指定父类,则默认继承Object类

子类从父类继承了哪些要素

  • 子类不继承父类的构造方法,但是总要调用父类的构造方法
  • 子类继承了父类的实例变量和实例方法、类变量和类方法,当然前提是没有被private修饰
  • 父类中被private修饰的成员,不会被子类继承
  • 父类中用final修饰的方法,子类不能重写

方法的重写override

  • 子类和父类拥有相同的方法签名,叫做子类重写了父类的方法
  • 两同两小一大原则:
    • 两同:方法名相同,形参列表相同
    • 两小:子类方法的返回值跟父类的相等或更小;子类方法抛出的异常要跟父类抛出的相等或更小
    • 一大:子类方法的访问权限要跟父类的相等或更大
    • static:子类方法要跟父类方法一致,要么是类方法,要么是实例方法,二者不能不一致
  • 方法重写后,子类对象将无法访问父类的方法,但可以在子类方法中调用父类的方法:super.实例方法,父类名.类方法
  • 如果父类方法是private修饰,则子类不能重写该方法;如果子类拥有一个跟父类private方法相同签名的方法,这也不是重写,这是子类新添加的方法,与父类的无关
  • 子类方法可能会和父类的方法发生重载
  • 父类中用final修饰的方法,子类不能重写

super关键字

  • 在子类中,用super关键字调用父类的实例变量或实例方法
  • super和static不能同时用来修饰方法;就像this不能和static不能同时修饰方法一样
  • 子类从父类继承了一个变量,又再定义了一个变量,而且同名,此时可以用super.变量名或者父类名.变量名来访问父类的这个变量

变量的查找顺序

  • 子类方法访问了一个变量,没有显式指定调用者的情况下,按以下顺序查找:
    • 当前方法中,是否有同名的局部变量
    • 当前类中,是否有同名的成员变量
    • 直接父类中,是否有
    • 逐级网上追溯,如果最终没有找到,那么提示编译出错

super调用父类构造器

  • 子类虽然不会继承父类的构造器,但总是会调用父类的构造器。调用同一个类的构造器用this,调用父类的用super
  • 显式调用:super调用语句写在在构造方法的第一行,因为this也要写在第一行,因此super和this不会同时出现
  • 间接调用:子类方法的第一行写的是this,被调用的构造器还得调用父类的构造器
  • 隐式调用:无super无this,系统默认调用父类的无参构造
  • 子类继承父类,总会一级一级的往上调用父类构造器,直到调用Object的,并且是找到顶部父类构造器后,开始往下一层一层执行
  • 比如下面的代码,继承关系:A-->B-->C-->D
public class Test{
public static void main(String[] args) {
D d=new D();
}
}
class D extends C{
D(){
System.out.println("D类构造器");
}
}
class C extends B{
C(){
System.out.println("C类构造器");
}
}
class B extends A{
B(){
System.out.println("B类构造器");
}
}
class A{
A(){
System.out.println("A类构造器");
}
}

输出:

A类构造器

B类构造器

C类构造器

D类构造器

  • 如果父类没有无参构造,子类又需要调用父类的无参构造,那么不能通过编译

多态Polymorphism

  • 简单的说,多态就是:在一个金字塔式的继承体系中,创建底部子类对象时,用顶端的父类类型指向这些底部的子类对象,通过相同类型的引用变量调用同一个方法时,会出现不同的结果
  • 多态源于:继承+父类型的引用指向子类型的对象+方法的重写
  • 引用变量有两个类型,一个是编译时类型,一个是运行时类型
  • 向上转型:父类型引用指向子类型对象,向上转型由系统自动完成
  • 向上转型的情况下:通过父类型的引用可以调用子类重写了的方法,但不能访问父类中没有子类中才有的实例变量,也就是说实例变量不具有多态性

instanceof与强制类型转换

  • 基本类型有强制类型转换,比如(double)16;引用类型也存在强制类型转换
  • 比如存在以下继承链:A-->B-->C-->D-->E-->F,B b=new D(),此时用B类型指向D对象,此时可以把b变量转为ACD类型,但不能转为EF类型,抛出异常ClassCastException
  • 进行强制类型转换之前,应先用instanceof进行判断能不能转,避免抛出异常
  • X变量 instanceof Y类型:判断X变量能不能转为Y类型,可能出现三种情况:
    • true表示可以转,意味着X变量指向的对象的类型,是Y类型或者Y的子类型;
    • false表示不可以转,意味着X变量的类型与Y类型存在父子继承关系,不可能是子父关系,X变量指向的对象的类型与Y类型没有父子或子父关系,二者都是X变量类型的子类型;
    • 编译不通过,表示X和Y没有父子或者子父继承关系

继承与组合

  • 继承可以实现代码的复用,但破坏了封装;组合也可以实现代码的复用
  • 设计父类遵循的原则:
    • 尽量用private隐藏父类的内部数据,不让子类直接访问父类的成员变量
    • 不要让子类随意访问、修改父类方法。父类辅助性的工具方法,用private修饰;需要让外部调用的方法,用public;不希望子类重写的方法用final修饰;希望子类重写而不希望外部访问则用protected
    • 不要在父类构造器中调用可能被子类重写的方法,后果很严重
  • 设计继承的原则:
    • 子类需要增加额外的属性,而不是属性的改变
    • 子类需要增加自己独有的功能或行为,可以增加新方法,或者重写父类方法
    • 不要光出于代码复用的目的设计继承,而要看实际情况,父类和子类是否有"has-a"的关系
  • 不想让一个类被继承:
    • final:用final修饰该类为最终类,不能被继承
    • private:用private修饰该类的所有构造方法,这样子类就无法调用该类的构造器,就不能继承了,另外应提供一个静态方法,用来返回该类的对象
  • 组合也能实现代码的复用
    • A类复用B类的方法
    • A类中创建B类的对象,并用private修饰
    • 在A类方法中调用B类对象的方法
  • 组合还是继承:
    • 继承:子类与父类是"is-a"关系,具体与抽象、特殊与一般的关系
    • 组合:新类与旧类是"has-a"关系,整体与局部的关系

其他:

  • 对于子类从父类继承了一个变量,又定义了一个同名变量的情况下,在创建子类对象时,为这两个变量都分配了内存,只是父类变量被隐藏