异常处理是我们日常开发中不可或缺的一部分。使用异常处理可以使我们的代码更加清晰规范,让我们能够快速定位到异常进行修正,使我们的程序更加健壮。除此之外还能给用户一个良好的出错提示。下面,我们将结合一个例子来谈谈异常处理机制。
1、示例
1 public class TestException { 2 public TestException() { 3 } 4 5 boolean testEx() throws Exception { 6 boolean ret = true; 7 try { 8 ret = testEx1(); 9 } catch (Exception e) { 10 System.out.println("testEx, catch exception"); 11 ret = false; 12 throw e; 13 } finally { 14 System.out.println("testEx, finally; return value=" + ret); 15 return ret; 16 } 17 } 18 19 boolean testEx1() throws Exception { 20 boolean ret = true; 21 try { 22 ret = testEx2(); 23 if (!ret) { 24 return false; 25 } 26 System.out.println("testEx1, at the end of try"); 27 return ret; 28 } catch (Exception e) { 29 System.out.println("testEx1, catch exception"); 30 ret = false; 31 throw e; 32 } finally { 33 System.out.println("testEx1, finally; return value=" + ret); 34 return ret; 35 } 36 } 37 38 boolean testEx2() throws Exception { 39 boolean ret = true; 40 try { 41 int b = 12; 42 int c; 43 for (int i = 2; i >= -2; i--) { 44 c = b / i; 45 System.out.println("i=" + i); 46 } 47 return true; 48 } catch (Exception e) { 49 System.out.println("testEx2, catch exception"); 50 ret = false; 51 throw e; 52 } finally { 53 System.out.println("testEx2, finally; return value=" + ret); 54 return ret; 55 } 56 } 57 58 public static void main(String[] args) { 59 TestException testException1 = new TestException(); 60 try { 61 testException1.testEx(); 62 } catch (Exception e) { 63 e.printStackTrace(); 64 } 65 } 66 }
上面的代码中,出现了异常的嵌套处理,看似很麻烦,但是只要我们掌握了异常处理的流程,就会很简单。
首先,testEx()的try块中调用了testEx1()方法,testEx1()的try块中又调用了testEx2()方法。
其次,在testtestEx2()中,for循环输出2,1。当 i=0 时,抛出异常,进入catch块,执行System.out.println("testEx2, catch exception"); ret = false; 进入finally块执行System.out.println("testEx2, finally; return value=" + ret); return ret; ret的值为false。
然后,跳回到testEx1()中,进入if语句满足条件,在return false 之前执行finally块中的语句,System.out.println("testEx1, finally; return value=" + ret); return ret; ret值为false
最后,跳回到testEx()中,try语句执行完成进入finally块,执行System.out.println("testEx, finally; return value=" + ret); return ret;
2、异常处理的流程
1)当try没有捕获到异常时:try语句块中的语句逐一被执行,程序将跳过catch语句块,执行finally语句块和其后的语句;
2)当try捕获到异常,catch语句块里没有处理此异常的情况:当try语句块里的某条语句出现异常时,而没有处理此异常的catch语句块时,此异常将会抛给JVM处理,finally语句块里的语句还是会被执行,但finally语句块后的语句不会被执行;
3)当try捕获到异常,catch语句块里有处理此异常的情况:在try语句块中是按照顺序来执行的,当执行到某一条语句出现异常时,程序将跳到 catch语句块,并与catch语句块逐一匹配,找到与之对应的处理程序,其他的catch语句块将不会被执行,而try语句块中,出现异常之后的语句 也不会被执行,catch语句块执行完后,执行finally语句块里的语句,最后执行finally语句块后的语句;
当涉及到finally语句时,我们应当注意:
1、无论是否捕获或处理异常,finally块里的语句都会被执行。
2、一旦try{ }中返回了某一个值,如果finally有返回值,finally中的返回值会覆盖try的返回值,如果finally没有返回值,就是try中的返回值。
3、如果try或catch块中有return语句或者throw语句,finally中的语句会在throw或return语句之前执行,但是在finally中对值的更新不会影响try catch块的返回结果。详情可以参考https://blog.csdn.net/abinge317/article/details/52253768
3、java中的异常
Java异常类层次结构图:
在 Java 中,所有的异常都有一个共同的祖先 Throwable(可抛出)。Throwable 指定代码中可用异常传播机制通过 Java 应用程序传输的任何问题的共性。Throwable有两个重要的子类:Exception(异常)和 Error(错误),二者都是 Java 异常处理的重要子类,各自都包含大量子类。
Exception和Error
Error(错误):是程序无法处理的错误,表示运行应用程序中较严重问题。大多数错误与代码编写者执行的操作无关,而表示代码运行时 JVM(Java 虚拟机)出现的问题。例如,Java虚拟机运行错误(Virtual MachineError),当 JVM 不再有继续执行操作所需的内存资源时,将出现 OutOfMemoryError。这些异常发生时,Java虚拟机(JVM)一般会选择线程终止。这些错误表示故障发生于虚拟机自身、或者发生在虚拟机试图执行应用时,如Java虚拟机运行错误(Virtual MachineError)、类定义错误(NoClassDefFoundError)等。这些错误是不可查的,因为它们在应用程序的控制和处理能力之 外,而且绝大多数是程序运行时不允许出现的状况。对于设计合理的应用程序来说,即使确实发生了错误,本质上也不应该试图去处理它所引起的异常状况。在 Java中,错误通过Error的子类描述。
Exception(异常):是程序本身可以处理的异常。
Exception 类有一个重要的子类 RuntimeException。RuntimeException 类及其子类表示“JVM 常用操作”引发的错误。例如,若试图使用空值对象引用、除数为零或数组越界,则分别引发运行时异常(NullPointerException、 ArithmeticException)和 ArrayIndexOutOfBoundException。
注意:
异常和错误的区别:异常能被程序本身可以处理,错误是无法处理。
可查异常与不可查异常
通常,Java的异常(包括Exception和Error)分为可查的异常(checked exceptions)和不可查的异常(unchecked exceptions)。
可查异常(编译器要求必须处置的异常):正确的程序在运行中,很容易出现的、情理可容的异常状况。可查异常虽然是异常状况,但在一定程度上它的发生是可以预计的,而且一旦发生这种异常状况,就必须采取某种方式进行处理。
不可查异常(编译器不要求强制处置的异常):包括运行时异常(RuntimeException与其子类)和错误(Error)。
除了RuntimeException及其子类以外,其他的Exception类及其子类都属于可查异常。这种异常的特点是Java编译器会检查它,也就 是说,当程序中可能出现这类异常,要么用try-catch语句捕获它,要么用throws子句声明抛出它,否则编译不会通过。
运行时异常和非运行时异常
(编译异常)。程序中应当尽可能去处理这些异常。
运行时异常:都是RuntimeException类及其子类异常,如NullPointerException(下标越界异常)等,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生。运行时异常的特点是Java编译器不会检查它,也就是说,当程序中可能出现这类异常,即使没有用try-catch语句捕获它,也没有用throws子句声明抛出它,也会编译通过。
非运行时异常 (编译异常):是 RuntimeException以外的异常,类型上都属于Exception类及其子类。从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不 能编译通过。如IOException、SQLException等以及用户自定义的Exception异常,一般情况下不自定义检查异常。
原文:https://www.cnblogs.com/wbringarden/p/9216964.html