这里所说的对象类型转换,是指存在继承关系的对象,不是任意类型的对象。当对不存在继承关系的对象进行强制类型转换时,java 运行时将抛出 java.lang.ClassCastException 异常。
在继承链中,我们将子类向父类转换称为“向上转型”,将父类向子类转换称为“向下转型”。
很多时候,我们会将变量定义为父类的类型,却引用子类的对象,这个过程就是向上转型。程序运行时通过动态绑定来实现对子类方法的调用,也就是多态性。在Java中由于继承和向上转型,子类可以非常自然地转换成父类。
然而有些时候为了完成某些父类没有的功能,我们需要将向下转型后的子类对象再转成子类,调用子类的方法,这就是向下转型。
因为子类拥有比父类更多的属性、更强的功能,所以父类转换为子类需要强制。那么,是不是只要是父类转换为子类就会成功呢?其实不然,他们之间的强制类型转换是有条件的。
当我们用一个类型的构造器构造出一个对象时,这个对象的类型就已经确定的,也就说它的本质是不会再发生变化了。在Java中我们可以通过继承、向上转型的关系使用父类类型来引用它,这个时候我们是使用功能较弱的类型引用功能较强的对象,这是可行的。但是将功能较弱的类型强制转功能较强的对象时,就不一定可以行了。
编译器在编译时只会检查类型之间是否存在继承关系,有则向下转型通过;而在运行时就会检查它的真实类型,如果真实类型确实是向下转型的类型则通过,否则抛出ClassCastException异常。
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashSet;
public class Solution {
public static void main( String[] args )
{
Collection list1 = new ArrayList();
HashSet set1 = (HashSet) list1;
}
}
Output :
Exception in thread “main” java.lang.ClassCastException: java.util.ArrayList cannot be cast to java.util.HashSet
at Solution.main(Solution.java:11)
正因为向下转型存在风险,所以在接收到父类的一个引用时,请务必使用 instanceof 运算符来判断该对象是否是你所要的子类。
参考文献:
http://blog.csdn.net/chenssy/article/details/14111307
http://www.weixueyuan.net/view/6002.html