Java Exception之多个catch

最近在温习Java基础的过程中遇到了异常处理中常见的try-catch问题

class ExceptionA extends Exception {}
public class ExceptionB extends ExceptionA {

    public static void main(String[] args) {

        try {
            throw new ExceptionB();
        } catch (ExceptionA e) {
            System.out.println("ExampleA");
        } catch (Exception e) {
            System.out.println("Exception");
        }
    }


相信很多人和我一样,认为结果是两个打印语句都执行,实际上执行结果如下

Java Exception之多个catch_第1张图片

首先解释打印ExampleA的原因:根据里氏代换原则[能使用父类型的地方一定能使用子类型],throw语句抛出的是子类ExceptionB异常,catch语句括号里的是父类ExceptionA的异常,因此可以捕获到。

后来查阅资料看到这样一段话:

对于try里面发生的异常,他会根据发生的异常和catch里面的进行匹配(怎么匹配,按照catch块从上往下匹配),当它匹配某一个catch块的时候,他就直接进入到这个catch块里面去了,后面在再有catch块的话,它不做任何处理,直接跳过去,全部忽略掉。

因此,在编码过程中对于多个catch的情况,每个catch的顺序要考虑好。


这个问题明白了,就来看下个问题

class Annoyance extends Exception {}

class Sneeze extends Annoyance {}

class Human {

    public static void main(String[] args) throws Exception {
        try {
            try {
                throw new Sneeze();
            } catch (Annoyance a) {
                System.out.println("Caught Annoyance");
                throw a;
            }
        } catch (Sneeze s) {
            System.out.println("Caught Sneeze");
        } finally {
            System.out.println("Hello World!");
        }
    }
}

这段代码的执行结果是

Java Exception之多个catch_第2张图片

很多人会疑惑为什么会打印Caught Sneeze,原因是内层catch里throw 的 a是其捕获的同层try跑出的Sneeze异常,也就是说a的类型是exception.Sneeze,因此外层catch可以捕获到。



你可能感兴趣的:(Java)