本小节我们来学习面向对象的最后一大特征——多态。多态是面向对象最重要的特性。我们将介绍多态的概念和特点,并带领大家实现一个多态的案例,你将了解到多态的实现条件什么是向上转型以及什么是向下转型,并学会使用instanceof运算符来检查对象引用是否是类型的实例。

1. 概念和特点

多态顾名思义就是多种形态,是指对象能够有多种形态。在面向对象中最常用的多态性发生在当父类引用指向子类对象时。在面向对象编程中,所谓多态意指相同的消息给予不同的对象会引发不同的动作。换句话说:多态意味着允许不同类的对象对同一消息做出不同的响应。
例如,火车类和飞机类都继承自交通工具类,这些类下都有各自的run()方法,交通工具的run()方法输出交通工具可以运输,而火车的run()方法输出火车会跑,飞机的run()方法则输出飞机会飞,火车和飞机都继承父类的run()方法,但是对于不同的对象,拥有不同的操作。
任何可以通过多个IS-A测试的 Java 对象都被视为多态的。在 Java 中,所有 Java 对象都是多态的,因为任何对象都能够通过IS-A测试以获取其自身类型和 Object 类。

2. 实现多态

2.1 实现条件

在 Java 中实现多态有 3 个必要条件:

  1. 满足继承关系
  2. 要有重写
  3. 父类引用指向子类对象

    2.1 实例

    例如,有三个类Pet、Dog、Cat:
    父类Pet ```java class Pet { // 定义方法 eat public void eat() {
    1. System.out.println("宠物吃东西");
    } }
  1. ```java
  2. //子类Dog继承Pet
  3. class Dog extends Pet { // 继承父类
  4. // 重写父类方法 eat
  5. public void eat() {
  6. System.out.println("狗狗吃狗粮");
  7. }
  8. }
  9. // 子类Cat继承Pet
  10. class Cat extends Pet { // 继承父类
  11. // 重写父类方法 eat
  12. public void eat() {
  13. System.out.println("猫猫吃猫粮");
  14. }
  15. }

在代码中,我们看到Dog和Cat类继承自Pet类,并且都重写了其eat方法。
现在已经满足了实现多态的前两个条件,那么如何让父类引用指向子类对象呢?我们在main方法中编写代码:

  1. public void main(String[] args) {
  2. // 分别实例化三个对象,并且保持其类型为父类Pet
  3. Pet pet = new Pet();
  4. Pet dog = new Dog();
  5. Pet cat = new Cat();
  6. // 调用对象下方法
  7. pet.eat();
  8. dog.eat();
  9. cat.eat();
  10. }

运行结果:

  1. 宠物吃东西
  2. 狗狗吃狗粮
  3. 猫猫吃猫粮

在代码中,Pet dog = new Dog();、Pet cat = new Cat();这两个语句,把Dog和Cat对象转换为Pet对象,这种把一个子类对象转型为父类对象的做法称为向上转型。父类引用指向了子类的实例。也就实现了多态。

2.3 向上转型

向上转型又称为自动转型、隐式转型。向上转型就是父类引用指向子类实例,也就是子类的对象可以赋值给父类对象。例如:

  1. Pet dog = new Dog();

这个是因为Dog类继承自Pet类,它拥有父类Pet的全部功能,所以如果Pet类型的变量指向了其子类Dog的实例,是不会出现问题的。
向上转型实际上是把一个子类型安全地变成了更加抽象的父类型,由于所有类的根类都是Object,我们也把子类类型转换为Object类型:

  1. Cat cat = new Cat();
  2. Object o = cat;

2.4 向下转型

向上转型是父类引用指向子类实例,那么如何让子类引用指向父类实例呢?使用向下转型就可以实现。向下转型也被称为强制类型转换。例如:

  1. // 为Cat类增加run方法
  2. class Cat extends Pet { // 继承父类
  3. // 重写父类方法 eat
  4. public void eat() {
  5. System.out.println("猫猫吃猫粮");
  6. }
  7. public void run() {
  8. System.out.println("猫猫跑步");
  9. }
  10. public static void main(String[] args) {
  11. // 实例化子类
  12. Pet cat = new Cat();
  13. // 强制类型转换,只有转换为Cat对象后,才能调用其下面的run方法
  14. Cat catObj = (Cat)cat;
  15. catObj.run();
  16. }
  17. }

运行结果:

  1. 猫猫跑步

我们为Cat类新增了一个run方法,此时我们无法通过Pet类型的cat实例调用到其下面特有的run方法,需要向下转型,通过(Cat)cat将Pet类型的对象强制转换为Cat类型,这个时候就可以调用run方法了。
使用向下转型的时候,要注意:不能将父类对象转换为子类类型,也不能将兄弟类对象相互转换。以下两种都是错误的做法:

  1. // 实例化父类
  2. Pet pet = new Pet();
  3. // 将父类转换为子类
  4. Cat cat = (Cat) pet;
  5. // 实例化Dog类
  6. Dog dog = new Dog();
  7. // 兄弟类转换
  8. Cat catObj = (Cat) dog;

不能将父类转换为子类,因为子类功能比父类多,多的功能无法凭空变出来。兄弟类之间不能转换,这就更容易理解了,兄弟类之间同样功能不尽相同,不同的功能也无法凭空变出来。

3. instanceof 运算符

instanceof运算符用来检查对象引用是否是类型的实例,或者这个类型的子类,并返回布尔值。如果是返回true,如果不是返回false。通常可以在运行时使用 instanceof 运算符指出某个对象是否满足一个特定类型的实例特征。其使用语法为:

  1. <对象引用> instanceof 特定类型

例如,在向下转型之前,可以使用instanceof运算符判断,这样可以提高向下转型的安全性:

  1. Pet pet = new Cat();
  2. if (pet instanceof Cat) {
  3. // 将父类转换为子类
  4. Cat cat = (Cat) pet;
  5. }

4. 小结

通过本小节的学习,我们知道了多态意味着一个对象有着多重特征,可以在特定的情况下,表现出不同状态,从而对应着不同的属性和方法。实现多态有 3 个必要条件,分别是要有继承要有重写以及父类引用指向子类对象,通过向上转型可以使父类引用指向子类实例;通过向下转型可以使子类引用指向父类实例,使用instanceof运算符可以用来检查对象引用是否是类型的实例。