Java多态


多态是同一个行为具有多个不同表现形式或者形态的能力。

多态就是一个接口,使用不同的实例而执行不同操作,如图所示:

 

 

 多态存在的三个必要条件

  • 继承
  • 重写
  • 父类引用指向子类对象

比如:

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("\n使用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
  • 方式二:接口
  • 方式三:抽象类和抽象方法

 

posted @ 2020-02-07 12:12  yl新蜜蜂  阅读(178)  评论(0编辑  收藏  举报