[TOC]
封装
将数据和基于数据的操作封装在一起,数据被保护在抽象数据类型的内部,只保留一些对外的接口使其与外部发生联系。
1 | public class Person { |
继承
继承实现了 IS-A 关系,例如 Cat 和 Animal 就是一种 IS-A 关系,因此 Cat 可以继承自 Animal,从而获得 Animal 非
private 的属性和方法。
继承应该遵循里氏替换原则,子类对象必须能够替换掉所有父类对象。
Cat 可以当做 Animal 来使用,也就是说可以使用 Animal 引用 Cat 对象。父类引用指向子类对象称为 向上转型 。
1 | Animal animal = new Cat(); |
多态
多态分为编译时多态和运行时多态:
编译时多态主要指方法的重载
运行时多态指的是 对象引用所指向的具体类型在运行期间才确定
运行时多态有三个条件:
1.继承
2.重写
3.向上转型
下面的代码中,乐器类(Instrument)有两个子类:Wind 和 Percussion,它们都覆盖了父类的 play() 方法,并且
在 main() 方法中使用父类 Instrument 来引用 Wind 和 Percussion 对象。在 Instrument 引用调用 play() 方法时,
会执行实际引用对象所在类的 play() 方法,而不是 Instrument 类的方法。
1 | public class Instrument { |
1 | public class Wind extends Instrument { |
1 | public class Percussion extends Instrument { |
1 | public class Music { |
1 | Wind is playing... |
重写与重载
重写(Override)
存在于继承体系中,指子类实现了一个与父类在方法声明上完全相同的一个方法。
为了满足里式替换原则,重写有以下三个限制:
1.子类方法的访问权限必须大于等于父类方法;
2.子类方法的返回类型必须是父类方法返回类型或为其子类型。
3.子类方法抛出的异常类型必须是父类抛出异常类型或为其子类型。
使用 @Override 注解,可以让编译器帮忙检查是否满足上面的三个限制条件。使用 @Override 注解,可以让编译器帮忙检查是否满足上面的三个限制条件。
在调用一个方法时,先从本类中查找看是否有对应的方法,如果没有查找到再到父类中查看,看是否有继承来的方
法。否则就要对参数进行转型,转成父类之后看是否有对应的方法。总的来说,方法调用的优先级为:
this.func(this)
super.func(this)
this.func(super)
super.func(super)