如何在Java中捕获多个异常

Java 7起支持在单个catch块中捕获多种异常,使用|分隔异常类型,适用于统一处理;但异常间不能存在继承关系,且具体异常需优先捕获以避免编译错误。

在Java中,捕获多个异常可以通过多种方式实现,目的是让代码更简洁、可读性更强。从Java 7开始,引入了对多异常捕获的改进,使得一个catch块可以处理多种不同类型的异常。

使用多重catch块(传统方式)

在早期版本的Java中,如果要处理多种异常,需要为每种异常写一个单独的catch块:

try {
    int result = 10 / Integer.parseInt("0");
    Object obj = null;
    obj.toString();
} catch (ArithmeticException e) {
    System.out.println("算术异常:" + e.getMessage());
} catch (NullPointerException e) {
    System.out.println("空指针异常:" + e.getMessage());
}

这种方式逻辑清晰,但当异常类型较多时,会导致代码冗长。

使用单一catch捕获多种异常(Java 7+)

Java 7引入了“多重异常捕获”语法,允许在一个catch块中捕获多种异常类型,使用|(或)操作符分隔:

try {
    int[] arr = new int[5];
    arr[10] = Integer.parseInt("abc");
} catch (ArrayIndexOutOfBoundsException | NumberFormatException e) {
    System.out.println("发生异常:" + e.getClass().getSimpleName());
}

这种写法减少了重复代码,适用于需要对多种异常做相同处理的情况。

注意:多个异常类型在catch中是并列关系,它们不能有继承关系(例如不能同时写Exception和RuntimeException),否则编译报错。

异常变量的类型与限制

在使用|操作符的多异常catch块中,异常参数的类型是这些异常的最小公共超类,但实际类型仍保持具体异常类型。

例如,NumberFormatException和ArrayIndexOutOfBoundsException都继承自RuntimeException,因此e的类型被视为RuntimeException,但你不能在catch块中修改它为更具体的类型。

限制:捕获的异常类之间不能存在继承关系,如下写法会编译失败:

// 编译错误:子类和父类不能同时出现在多catch中
catch (Exception | NullPointerException e)

结合不同处理逻辑的策略

如果不同异常需要不同的处理方式,应使用多个catch块,并将更具体的异常放在前面:

try {
    parseAndAccess();
} catch (NumberFormatException e) {
    System.out.println("数字格式错误");
} catch (IllegalArgumentException e) {
    System.out.println("非法参数");
} catch (Exception e) {
    System.out.println("其他异常");
}

JVM会按顺序匹配异常,因此具体异常必须放在通用异常之前,否则无法编译。

基本上就这些。合理使用多异常捕获能让代码更干净,但也要根据实际处理需求选择合适的方式。不复杂但容易忽略的是异常的继承关系和捕获顺序。