多态
1、基础语法
向上转型(upcasting)
子类型–>父类型
又被称为:自动类型转换
向下转型(downcasting)
父类型–>子类型
又被称为:强制类型转换。【需要加强制类型转换符】
无论是向上转型还是向下转型,两种类型之间必须要有继承关系。
没有继承关系,程序是无法编译通过的。
使用多态语法机制
1.Animal和Cat之间存在继承关系,Animal是父类,Cat是子类
2.Cat is a Animal
3.new Cat()创建的对象的类型是Cat,a2这个引用的数据类型是Animal,可见它们进行了类型转换
子类型转化为父类型,称为向上转型,或者称为自动类型转换。
4.Java中允许这种语法:父类型引用指向子类型对象。
Aniaml a1 = new Animal();
Aniaml a2 = new Cat();
a1.move(); //动物在移动
a2.move();// 猫在走猫步
//a2.catchMouse(); //编译不通过,Animal类中没有catchMouse方法
//因为编译阶段编译器检查a2的类型是Animal类型,
//从Animal.class字节码文件当中查找catchMouse()方法,
//最终没有找到该方法,导致静态绑定失败,没有绑定成功,也就是说编译失败了。
- java程序永远都分为编译阶段和运行阶段
- 先分析编译阶段,在分析运行阶段,编译无法通过,根本是无法运行的。
- 编译阶段编译器检查a2这个引用的数据类型为Animal,由于Animal.class字节码当中有move()方法,所以编译通过了。这个过程我们称为静态绑定,编译阶段绑定。只有静态绑定成功了才有后续的运行。
- 在程序运行阶段,JVM堆内存当中真实创建的对象是Cat对象,那么以下程序在运行阶段一定会调用Cat对象的move()方法,此时发生了程序的动态绑定,运行阶段绑定。
- 无论是Cat类有没有重写move的方法,运行阶段一定调用的是Cat对象的move方法,因为底层真实对象就是Cat对象
- 父类型引用指向子类型对象这种机制导致程序存在编译阶段绑定和运行阶段绑定两种不同的形态/状态,这种机制可以称为一种多态语法机制。
Animal a3 = new Bird();
/**
* 1.以下程序编译是没有问题的,因为编译器检查到a3的数据类型是Animal
* Animal和Cat之间存在继承关系,并且Anima1是父类型,Cat是子类型,
*父类型转换成子类型叫做向下转型,语法合格。
* 2、程序虽然编译通过了,但是程序在运行阶段会出现异常,因为JVM雄内存
*当中真实存在的对象是Bi rd类型, Bi rd对象无法转换成Cat对象,因为两种
*类型之间不存在任何继承关系,此时出现了著名的异常:java.lang . ClassCastException
类型转换异常,这种异常总是在“向下转型"的时候会发生。
*/
Cat c3 = (Cat)a3;
怎么解决 ClassCastException异常呢?
使用instanceof运算符可以避免出现以上异常。
instanceof运算符怎么使用?
语法格式:
引用 instanceof 数据类型名
以上运算符的执行结果类型是布尔类型,结果可能是true/false
关于运算结果true/false:
假设:(a instanceof Animal)
- true 表示:a这个引用指向的对象是一个Animal类型
- false表示:a这个引用指向的对象不是一个Animal类型。
Java规范中要求:在进行强制类型转换之前,建议采用instanceof运算符进行判断,避免ClassCastException异常的发生。 这是一种好习惯。
if (a3 instanceof Cat){ //a3是一个Cat类型的对象
Cat c3 = (Cat)a3;
c3.catchMouse();
}else if(a3 instanceof Bird){//a3是一个Bird类型的对象
Bird b2 = (Bird)a3;
b2.fly();
}
}
2、多态在实际开发中的作用
-
- 降低耦合度【解耦合】,提高程序的扩展力【软件开发的一个很重要的目标】
-
- 能使用多态尽量使用多态
-
- 父类型引用指向子类型对象
明天休息一天嘻嘻嘻~~