Java中Downcasting向下转型的用处

看如下代码:

class Animal {
	void makeNoise() {System.out.println("普通叫声"); }
}
class Dog extends Animal {
	void makeNoise() {System.out.println("汪汪汪"); }
	void playDead() { System.out.println("翻滚"); }
}
class Test{
	public static void main(String [] args) {
		Animal [] a = {new Animal(), new Dog(), new Animal() };
		for(Animal animal : a) {
		animal.makeNoise();
		if(animal instanceof Dog) {
			animal.playDead(); //代码段A
		}
	}
}

循环Animal实例时我需要判断当它是狗的时候再调用playDead方法,但是上述代码会包编译错误

The method playDead() is undefined for the type Animal

因为编译的时候都是Animal对象,没有子类Dog中的playDead方法,所以需要进行向下转型Downcast,将代码片段A改为如下代码:

Dog d = (Dog) animal;
d.playDead();

就会正常运行:

普通叫声
汪汪汪
翻滚
普通叫声

 虽然downcast在用于特定的子类是特别有用,但是不能一味的进行downcast:

class Animal { }
class Dog extends Animal { }
class DogTest {
	public static void main(String [] args) {
		Animal animal = new Animal();
		Dog d = (Dog) animal; 
	}
}
该代码能编译但是在运行时会出错:

Exception in thread "main" java.lang.ClassCastException: Animal cannot be cast to Dog
	at DogTest.main(Animal.java:7)


因为Dog继承Animal,所以编译器认为这些Animal对象都有可能是Dog,所以编译不会报错,除非编译器能很明确的知道downcast不行,比如:

Animal animal = new Animal();
String d = (String) animal; 

这段代码在编译时就会报错,因为编译器能很明确知道Animal和String不在同一个继承关系树上。

你可能感兴趣的:(Java基础,向下转型,java,downcast)