异常体系
上图为异常机制的结构图
Throwable是异常的父类,它的两个子类Exception(异常)和 Error(错误),二者都是 Java 异常处理的重要子类,各自都包含大量子类。
错误是程序无法处理的错误,表示运行应用程序中较严重问题,多为硬件故障,程序本身不能解决;
异常是程序本身可以处理的异常, Exception 类有一个重要的子类 RuntimeException。
运行异常(编译异常): 都是RuntimeException类及其子类异常,如NullPointerException(空指针异常)、IndexOutOfBoundsException(下标越界异常)等,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生。
受检异常: 是RuntimeException以外的异常,类型上都属于Exception类及其子类。从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过。如IOException、SQLException等以及用户自定义的Exception异常,一般情况下不自定义检查异常
异常处理机制:
在 Java 应用程序中,异常处理机制为:抛出异常,捕捉异常。
抛出异常:当一个方法出现错误引发异常时,方法创建异常对象并交付运行时系统,异常对象中包含了异常类型和异常出现时的程序状态等异常信息。运行时系统负责寻找处置异常的代码并执行。
捕获异常:在方法抛出异常之后,运行时系统将转为寻找合适的异常处理器(exception handler)。潜在的异常处理器是异常发生时依次存留在调用栈中的方法的集合。当异常处理器所能处理的异常类型与方法抛出的异常类型相符时,即为合适 的异常处理器。运行时系统从发生异常的方法开始,依次回查调用栈中的方法,直至找到含有合适异常处理器的方法并执行。当运行时系统遍历调用栈而未找到合适 的异常处理器,则运行时系统终止。同时,意味着Java程序的终止。
*
任何抛出异常都必须使用throws子句。
捕捉异常通过try-catch语句或者try-catch-finally语句实现。
*
捕捉异常:
1. try catch语句
一般语法形式:
try {
// 可能会发生异常的程序代码
} catch (异常类型一 异常变量一){
// 捕获并处置try抛出的异常类型一
}
catch (异常类型二 异常变量二){
//捕获并处置try抛出的异常类型二
}
……
匹配的原则是:如果抛出的异常对象属于catch子句的异常类,或者属于该异常类的子类,则认为生成的异常对象与catch块捕获的异常类型相匹配。
注意::一旦某个catch捕获到匹配的异常类型,将进入异常处理代码。一经处理结束,就意味着整个try-catch语句结束。其他的catch子句不再有匹配和捕获异常类型的机会。
2. try catch finally语句
一般语法形式:
try {
// 可能会发生异常的程序代码
} catch (异常类型一 异常变量一){
// 捕获并处置try抛出的异常类型一
}
catch (异常类型二 异常变量二){
//捕获并处置try抛出的异常类型二
}finally{
//无论是否异常出现都执行的语句块
}
总结:
try块:用于捕获异常。其后可接零个或多个catch块,如果没有catch块,则必须跟一个finally块。
catch块:用于处理try捕获到的异常。
finally块:无论是否捕获或处理异常,finally块里的语句都会被执行。当在try块或catch块中遇到return语句时,finally语句块将在方法返回之前被执行。
注意:
catch关键字后面括号中的Exception类型的参数e。Exception就是try代码块传递给catch代码块的变量类型,e就是变量名。catch代码块中语句”e.getMessage();”用于输出错误性质。通常异常处理常用3个函数来获取异常的有关信息:
getCause():返回抛出异常的原因。如果 cause 不存在或未知,则返回 null。
getMeage():返回异常的消息信息。
printStackTrace():对象的堆栈跟踪输出至错误输出流,作为字段 System.err 的值。
*::: 什么时候能使finally语句块不执行,资料查询:在以下4种特殊情况下,finally块不会被执行:
1)在finally语句块中发生了异常。
2)在前面的代码中用了System.exit()退出程序。
3)程序所在的线程死亡。
4)关闭CPU。*
3. try、catch、finally语句块的执行顺序:
1)当try没有捕获到异常时:逐一执行try语句块中的语句,跳过catch语句块,执行finally语句块和其后的语句;
2)当try捕获到异常,但是catch语句块中并没有没有处理此异常的情况:异常将会抛给JVM处理,并执行finally语句块里的语句,但finally语句块后的语句不会被执行;
3)当try捕获到异常,catch语句块中有处理此异常的情况:顺序执行try语句块,当执行到出现异常的某一条语句时,程序将跳到catch语句块,并与catch语句块逐一匹配,找到与之对应的处理程序,其他的catch语句块将不会被执行,与此同时try语句块中,出现异常之后的语句也不会被执行,catch语句块执行完后,执行finally语句块里的语句,最后执行finally语句块后的语句;
抛出异常:
throws和throw一起使用
throws语句用在方法定义时声明该方法要抛出的异常类型
throw总是出现在函数体中,用来抛出一个Throwable类型的异常
………………………………………………(未完待续)