ClassCastException 是 Java 中常见的运行时异常,它通常发生在试图将一个对象强制转换为不兼容的类型时。这种转换在编译时是合法的,但在运行时会检查对象的实际类型,如果与目标类型不匹配,就会抛出 ClassCastException。
以下是可能导致 ClassCastException 的一些常见情况:
直接实例化接口或抽象类:
interface MyInterface { }
public class Main {
public static void main(String[] args) {
MyInterface obj = new MyInterface() {}; // 编译通过,但运行时抛出 ClassCastException
}
}
在这个例子中,尽管我们创建了一个匿名内部类实现了 MyInterface,但直接实例化接口本身会导致 ClassCastException。
错误的类型转换:
List<String> stringList = new ArrayList<>();
stringList.add("Hello");
List<Integer> integerList = (List<Integer>) stringList; // 运行时抛出 ClassCastException
这里我们试图将一个 List<String> 强制转换为 List<Integer>,这是不合法的,因为列表中的元素类型不匹配。
数组类型转换:
Object[] objects = new String[1];
Integer[] integers = (Integer[]) objects; // 运行时抛出 ClassCastException
在这个例子中,尽管 objects 数组可以容纳一个对象,但我们试图将其强制转换为 Integer[] 数组,这会导致 ClassCastException。
泛型类型擦除: 由于 Java 的泛型是类型擦除的,因此在运行时无法检查泛型的实际类型。例如:
List<String> stringList = new ArrayList<>();
List rawList = stringList;
rawList.add(1); // 编译通过,但运行时可能导致 ClassCastException
尽管 rawList 是 List 类型的,但由于类型擦除,无法确保它只包含 String 对象。
要避免 ClassCastException,可以在强制转换之前进行类型检查,或者使用更安全的泛型操作。例如:
List<String> stringList = new ArrayList<>();
if (stringList instanceof List<?> && ((List<?>) stringList).isEmpty()) {
List<?> rawList = stringList;
rawList.add(1); // 这里不会抛出 ClassCastException,因为进行了类型检查
}