捕获异常

在Java中,凡是可能抛出异常的语句,都可以用try … catch捕获。把可能发生异常的语句放在try { … }中,然后使用catch捕获对应的Exception及其子类。

多catch语句

可以使用多个catch语句,每个catch分别捕获对应的Exception及其子类。JVM在捕获到异常后,会从上到下匹配catch语句,匹配到某个catch后,执行catch代码块,然后不再继续匹配。

简单地说就是:多个catch语句只有一个能被执行。例如:

  1. public static void main(String[] args) {
  2. try {
  3. process1();
  4. process2();
  5. process3();
  6. } catch (IOException e) {
  7. System.out.println(e);
  8. } catch (NumberFormatException e) {
  9. System.out.println(e);
  10. }
  11. }

存在多个catch的时候,catch的顺序非常重要:子类必须写在前面。例如:

  1. public static void main(String[] args) {
  2. try {
  3. process1();
  4. process2();
  5. process3();
  6. } catch (IOException e) {
  7. System.out.println("IO error");
  8. } catch (UnsupportedEncodingException e) { // 永远捕获不到
  9. System.out.println("Bad encoding");
  10. }
  11. }

对于上面的代码,UnsupportedEncodingException异常是永远捕获不到的,因为它是IOException的子类。当抛出UnsupportedEncodingException异常时,会被catch (IOException e) { … }捕获并执行。

因此,正确的写法是把子类放到前面:

  1. public static void main(String[] args) {
  2. try {
  3. process1();
  4. process2();
  5. process3();
  6. } catch (UnsupportedEncodingException e) {
  7. System.out.println("Bad encoding");
  8. } catch (IOException e) {
  9. System.out.println("IO error");
  10. }
  11. }

finally语句

无论是否有异常发生,如果我们都希望执行一些语句,例如清理工作,怎么写?

可以把执行语句写若干遍:正常执行的放到try中,每个catch再写一遍。例如:

  1. public static void main(String[] args) {
  2. try {
  3. process1();
  4. process2();
  5. process3();
  6. System.out.println("END");
  7. } catch (UnsupportedEncodingException e) {
  8. System.out.println("Bad encoding");
  9. System.out.println("END");
  10. } catch (IOException e) {
  11. System.out.println("IO error");
  12. System.out.println("END");
  13. }
  14. }

上述代码无论是否发生异常,都会执行System.out.println("END");这条语句。

那么如何消除这些重复的代码?Java的try … catch机制还提供了finally语句,finally语句块保证有无错误都会执行。上述代码可以改写如下:

  1. public static void main(String[] args) {
  2. try {
  3. process1();
  4. process2();
  5. process3();
  6. } catch (UnsupportedEncodingException e) {
  7. System.out.println("Bad encoding");
  8. } catch (IOException e) {
  9. System.out.println("IO error");
  10. } finally {
  11. System.out.println("END");
  12. }
  13. }

注意finally有几个特点:

  • finally语句不是必须的,可写可不写;
  • finally总是最后执行。如果没有发生异常,就正常执行try { … }语句块,然后执行finally。如果发生了异常,就中断执行try { … }语句块,然后跳转执行匹配的catch语句块,最后执行finally

可见,finally是用来保证一些代码必须执行的。

某些情况下,可以没有catch,只使用try … finally结构。例如:

  1. void process(String file) throws IOException {
  2. try {
  3. ...
  4. } finally {
  5. System.out.println("END");
  6. }
  7. }

因为方法声明了可能抛出的异常,所以可以不写catch

捕获多种异常

如果某些异常的处理逻辑相同,但是异常本身不存在继承关系,那么就得编写多条catch子句:

  1. public static void main(String[] args) {
  2. try {
  3. process1();
  4. process2();
  5. process3();
  6. } catch (IOException e) {
  7. System.out.println("Bad input");
  8. } catch (NumberFormatException e) {
  9. System.out.println("Bad input");
  10. } catch (Exception e) {
  11. System.out.println("Unknown error");
  12. }
  13. }

因为处理IOExceptionNumberFormatException的代码是相同的,所以我们可以把它两用|合并到一起:

  1. public static void main(String[] args) {
  2. try {
  3. process1();
  4. process2();
  5. process3();
  6. } catch (IOException | NumberFormatException e) { // IOException或NumberFormatException
  7. System.out.println("Bad input");
  8. } catch (Exception e) {
  9. System.out.println("Unknown error");
  10. }
  11. }

练习

try … catch捕获异常并处理。

下载练习:捕获异常练习 (推荐使用IDE练习插件快速下载)

小结

使用try … catch … finally时:

  • 多个catch语句的匹配顺序非常重要,子类必须放在前面;

  • finally语句保证了有无异常都会执行,它是可选的;

  • 一个catch语句也可以匹配多个非继承关系的异常。

读后有收获可以支付宝请作者喝咖啡,读后有疑问请加微信群讨论

捕获异常 - 图1捕获异常 - 图2