通俗来讲,就是多种形态。就是完成某一个行为时,如果不同对象去完成就会产生不同的状态。
总的来讲,同一件事情,发生在不同对象的身上,就会产生不同的结果。
下面们,我们来详细解析下讲下这三个条件
这里我们创建 一个父类 Animal 和 两个子类 Dog 和 Cat
class Animal{
public String name;
public int age;
public Animal(String name,int age){ // 带有两个参数的构造方法
this.name = name;
this.age = age;
}
public void eat(){
System.out.println("吃饭");
}
}
class Dog extends Animal{
public Dog(String name ,int age){
super(name,age);
}
@Override
public void eat() {
System.out.println("吃狗粮");
}
}
class Cat extends Animal{
public Cat(String name ,int age){
super(name,age);
}
@Override
public void eat() {
System.out.println("吃猫粮");
}
}
Dog类重写 eat() 方法
Cat类重写 eat() 方法
在测试类中,我们通过父类的引用了重写的方法
public class Test {
public static void func(Animal animal){
animal.eat();
}
public static void main(String[] args) {
Animal animal1 = new Dog("大黄",1);
func(animal1);
Animal animal2 = new Cat("喵喵",2);
animal2.eat(); // 通过父类的引用 调用重写方法
}
}
最终的输出结果就是:
我们来画个图分析下, 下面这个例子就形成了多态的条件
那么,程序是如何运行的呢?我们继续来看一个图,来分析下
在将 class文件,进行反汇编以后,也可以看出 程序在编译的时候,也是调用的 父类的 eat() 方法
而在运行的时候,却是调用了重写以后的方法,这就是动态绑定,也是发生多态的基础!
重写(override):也称为覆盖。重写是子类对父类非静态、非private修饰,非final修饰,非构造方法等的实现过程进行重新编写。 有以下几条规则
既然讲到重写,那么我们就来对比下重写和重载的区别
方法的重载是一个类的多态的表现
方法重写是子类和父类的一种多态表现
创建一个子类对象,将其当成父类对象来使用
语法格式: 父类类型 对象名 = new 子类类型()
Animal animal = new Cat();
使用场景有三种:
Animal animal = new Cat("旺财",5);
public class Test {
public static void func(Animal animal){
animal.eat();
}
public static void main(String[] args) {
Animal animal1 = new Dog("大黄",1);
Animal animal2 = new Cat("喵喵",2);
func(animal1);
func(animal2);
}
}
public static Animal funciton(String name){
return new Dog("大黄",2);
}
向上转型的优点:让代码更加灵活,可以引用更多的类型
向上转型的缺点:不能调用子类特有的方法(除非重写,发生动态绑定)
语法格式:
父类类型 父类对象名 = new 子类类型()
子类类型 子类对象名 = (子类类型)父类对象名
Animal animal1 = new Dog("黑咯破",1);
Dog dog = (Dog) animal1;
向下转型用的比较少,不安全,可以从下图中看出不安全。
Java 为了提高向下转型的安全性,引入 instance关键字,如果该表达式 为 true,则可以安全转换。
好处:
多态也有缺陷:
我们要避免在构造方法中调用重写的方法
class B {
public B() {
// do nothing
func();
}
public void func() {
System.out.println("B.func()");
}
}
class D extends B {
private int num = 1;
@Override
public void func() {
System.out.println("D.func() " + num);
}
}
public class Test {
public static void main(String[] args) {
D d = new D();
}
}
上述代码执行的结果是 D.fun() 0 , 通过下图来看执行步骤
当在父类的构造方法中 去调用父类和子类的重写方法时,此时会调用子类的!