[Java] 语言允许某个类型的引用变量引用子类的实例,而且可以对这个引用变量进行类型转换。如果把引用类型转换为子类类型,则称为向下转型;如果把引用类型转换为父类类型,则称为向上转型。
例如,Creature 类表示生物类,Animal 类表示动物类,该类对应的子类有 Dog 类,使用对象类型表示如下:
1. Animal animal = new Dog();
2. Dogdog = (Dog) animal; // 向下转型,把Animal类型转换为Dog类型
3. Creature creature = animal; // 向上转型,把Animal类型转换为Creature类型
例 1
下面通过具体的示例演示对象类型的转换。例如,父类 Animal 和子类 Cat 中都定义了实例变量 name、静态变量 staticName、实例方法 eat() 和静态方法 staticEat()。此外,子类 Cat 中还定义了实例变量 str 和实例方法 dogMethod()。
父类 Animal 的代码如下:
1. public class Animal {
2. public String name = "Animal:动物";
3. public static String staticName = "Animal:可爱的动物";
5. public void eat() {
6. System.out.println("Animal:吃饭");
7. }
9. public static void staticEat() {
10. System.out.println("Animal:动物在吃饭");
11. }
12. }
子类 Cat 的代码如下:
1. public class Cat extends Animal {
2. public String name = "Cat:猫";
3. public String str = "Cat:可爱的小猫";
4. public static String staticName = "Dog:我是喵星人";
6. public void eat() {
7. System.out.println("Cat:吃饭");
8. }
10. public static void staticEat() {
11. System.out.println("Cat:猫在吃饭");
12. }
14. public void eatMethod() {
15. System.out.println("Cat:猫喜欢吃鱼");
16. }
18. public static void main(String[] args) {
19. Animal animal = new Cat();
20. Cat cat = (Cat) animal; // 向下转型
21. System.out.println(animal.name); // 输出Animal类的name变量
22. System.out.println(animal.staticName); // 输出Animal类的staticName变量
23. animal.eat(); // 输出Cat类的eat()方法
24. animal.staticEat(); // 输出Animal类的staticEat()方法
25. System.out.println(cat.str); // 调用Cat类的str变量
26. cat.eatMethod(); // 调用Cat类的eatMethod()方法
27. }
28. }
通过引用类型变量来访问所引用对象的属性和方法时,Java 虚拟机将采用以下绑定规则:
- 实例方法与引用变量实际引用的对象的方法进行绑定,这种绑定属于动态绑定,因为是在运行时由 Java 虚拟机动态决定的。例如,animal.eat() 是将 eat() 方法与 Cat 类绑定。
- 静态方法与引用变量所声明的类型的方法绑定,这种绑定属于静态绑定,因为是在编译阶段已经做了绑定。例如,animal.staticEat() 是将 staticEat() 方法与 Animal 类进行绑定。
- 成员变量(包括静态变量和实例变量)与引用变量所声明的类型的成员变量绑定,这种绑定属于静态绑定,因为在编译阶段已经做了绑定。例如,animal.name 和 animal.staticName 都是与 Animal 类进行绑定。在此我向大家推荐一个架构学习交流圈。交流学习指导伪鑫:1253431195(里面有大量的面试题及答案)里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化、分布式架构等这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多
对于 Cat 类,运行时将会输出如下结果:
Animal:动物 Animal:可爱的动物 Cat:吃饭 Animal:动物在吃饭 Cat:可爱的小猫 Cat:猫喜欢吃鱼
强制对象类型转换
Java 编译器允许在具有直接或间接继承关系的类之间进行类型转换。对于向下转型,必须进行强制类型转换;对于向上转型,不必使用强制类型转换。
例如,对于一个引用类型的变量,Java 编译器按照它声明的类型来处理。如果使用 animal 调用 str 和 eatMethod() 方法将会出错,如下:
1. animal.str = ""; // 编译出错,提示Animal类中没有str属性
2. animal.eatMethod(); // 编译出错,提示Animal类中没有eatMethod()方法
如果要访问 Cat 类的成员,必须通过强制类型转换,如下:
1. ((Cat)animal).str = ""; // 编译成功
2. ((Cat)animal).eatMethod(); // 编译成功
把 Animal 对象类型强制转换为 Cat 对象类型,这时上面两句编译成功。对于如下语句,由于使用了强制类型转换,所以也会编译成功,例如:
1. Cat cat = (Cat)animal; // 编译成功,将Animal对象类型强制转换为Cat对象类型
子类的对象可以转换成父类类型,而父类的对象实际上无法转换为子类类型。因为通俗地讲,父类拥有的成员子类肯定也有,而子类拥有的成员,父类不一定有。因此,对于向上转型,不必使用强制类型转换。例如:
1. Cat cat = new Cat();
2. Animal animal = cat; // 向上转型,不必使用强制类型转换
如果两种类型之间没有继承关系,那么将不允许进行类型转换。例如:
1. Dog dog = new Dog();
2. Cat cat = (Cat)dog; // 编译出错,不允许把Dog对象类型转换为Cat对象类型