多态是同一个行为具有多个不同表现形式或者形态的能力。
多态就是一个接口,使用不同的实例而执行不同操作,如图所示:
多态存在的三个必要条件
- 继承
- 重写
- 父类引用指向子类对象
比如:
Parent p=new Child();
当使用多态方式调用方法时,首先检查父类中是否有该方法,如果没有,则编译错误;如果有,再去调用子类的同名方法。
多态的好处:可以使程序有良好的扩展,并可以对所有类的对象进行通用处理。
package pkg2020华南虎; /** * * @author yl */ public class TestDuoTai { public static void main(String[] args) { show(new Cat());//Cat对象调用show方法 show(new Dog00());//Dog对象调用show方法 Animal00 a = new Cat();//向上转型 a.eat();//调用Cat的eat方法 Cat c = (Cat) a;//向下转型 c.work(); } public static void show(Animal00 a) { a.eat(); if (a instanceof Cat) { Cat c = (Cat) a; c.work(); } else if (a instanceof Dog00) { Dog00 c = (Dog00) a; c.work(); } } } abstract class Animal00 { abstract void eat(); } class Cat extends Animal00 { public void eat() { System.out.println("吃鱼"); } public void work() { System.out.println("抓老鼠"); } } class Dog00 extends Animal00 { public void eat() { System.out.println("吃骨头"); } public void work() { System.out.println("看家"); } }
虚函数
虚函数的存在是为了多态。
Java中其实没有虚函数的概念,他的普通函数就相当于C++的虚函数,动态绑定是Java的默认行为。如果Java中不希望某个函数具有虚函数特性,可以加上final关键字变成非虚函数。
重写
我们将介绍在Java中,当设计类时,被重写的方法的行为怎样影响多态性。
我们已经讨论了方法的重写,也就是子类能够重写父类的方法。
当子类对象调用重写方法时,调用的是子类的方法,而不是父类中被重写的方法。
要想调用父类中被重写的方法,则必须使用关键字super。
package pkg2020华南虎; /** * * @author yl */ public class EmployeeTest { private String name; private String address; private int number; public EmployeeTest(String name, String address, int number) { System.out.println("Employee构造函数"); this.name = name; this.address = address; this.number = number; } public void mailCheck() { System.out.println("邮寄支票给:" + this.name + " " + this.address); } @Override public String toString() { return "EmployeeTest{" + "name=" + name + ", address=" + address + ", number=" + number + '}'; } public String getName() { return name; } public String getAddress() { return address; } public int getNumber() { return number; } public void setAddress(String address) { this.address = address; } }
package pkg2020华南虎; /** * * @author yl */ public class Salary extends EmployeeTest { private double salary;//全年工资 public Salary(String name, String address, int number, double salary) { super(name,address,number); setSalary(salary); } public void mailCheck(){ System.out.println("Salary类的mailCheck方法"); System.out.println("邮寄支票给:"+getName()+",工资为:"+salary); } public double getSalary() { return salary; } public void setSalary(double newsalary) { if(newsalary>=0.0) this.salary = newsalary; } public double computePay(){ System.out.println("JI算工资,付给:"+getName()); return salary/52; } }
package pkg2020华南虎; /** * * @author yl */ public class VirtualDemo { public static void main(String[] args) { Salary s=new Salary("员工A","北京",3,8888); EmployeeTest e=new Salary("员工B","上海",2,6666); System.out.println("使用Salary的引用调用mailCheck---"); s.mailCheck(); System.out.println(" 使用EmployeeTest的引用调用mailCheck---"); e.mailCheck(); } }
例子解析
- 实例中,实例化了两个Salary对象:一个使用Salary引用s,另一个使用EmployeeTest引用e。
- 当调用s.mailCheck()时,编译器在编译时会在Salary类中找到mailCheck(),执行过程JVM就调用Salary类的买了Check()。
- 因为e是EmployeeTest的引用,所以调用e的mailCheck()方法时,编译器会去EmployeeTest类查找mailCheck()方法。
- 在编译的时候,编译器使用EmployeeTest类中的mailCheck()方法验证该语句,但是在运行的时候,JVM调用的是Salary类中的mailCheck()方法。
以上整个过程被称为虚拟方法调用,该方法被称为虚拟方法。
Java中所有的方法都能以这种方式表现,因此,重写的方法能在运行时调用,不管编译的时候源代码中引用变量是什么数据类型。
多态的实现方式
- 方式一:重写Override
- 方式二:接口
- 方式三:抽象类和抽象方法