1.异常分类
按继承关系分
异常可以分为Error和Exception类;其中,Exception类又可以分为RuntimeException类和非RuntimeException类(如IOException类)。
按受查与否分
Error类和Exception类中的RuntimeException类属于非受查异常,Exception类下的其他非RuntimeException类属于受查异常。
2.异常的抛出throw
- 一般我们编写的都是受查异常,也就是说会在编译的时候接受编译器的检查。对于受查异常,为了满足编译器检查,我们要么继续抛出(声明)该异常,要么着手解决该异常。
- 若方法内部会抛出(throw)异常,并且该异常没有得到解决,则在编写该方法时,就要在方法名后声明(throws)其包含的异常,多个异常用逗号隔开。
3.异常的捕获
异常的捕获处理可以按照下述进行:
try{
··
throw new ExceptionTypeA();
··
}
catch(ExceptionTypeA | ExceptionTypeB e){//1)
code
}
catch(ExceptionTypeC e){
code
}
finally{
code//常为关闭IO流操作
}
- catch可以捕获多个异常类进行相同操作,如1)处所示
- 异常隐含为final变量,对异常变量的补充常采用异常链形式,可以捕获当前异常,处理,然后新建一个异常,将新异常的原因设置为捕获的异常。详见P274
- catch后可以加finally,finally用于无论try和catch怎么执行,最终都要执行finall语句块。并且,在try-catch-finally语句块中,finally语句块中的return/抛出异常(立即结束语句)的优先级最高,程序会优先返回finally语句块中的立即结束语句的结果,此时try-catch语句块中的return/抛出异常(立即结束语句)的结果就会被丢弃掉。所以我们在将捕获的异常抛出给调用的上层方法处理时,如果被finally语句块中的return语句覆盖掉了,那么这个导致异常的错误情况将很难发现和定位。——避免在 finally 语句块中使用 return 语句
- 当使用finally语句块来进行自动关闭IO操作时,可以改用带资源的try语句,详见P278
4.堆栈轨迹的分析
堆栈轨迹的分析常用的有三种方法,要注意的是分析的方法全部继承于Throwable类,非Exception类所有,具体如下
4.1.printStackTrace方法
采用如下调用:
catch(Exception e){
e.printStackTrace()
}
该方法会打印异常的轨迹信息。
4.2.getStackTrace方法
该方法声明如下
public StackTraceElement[] getStackTrace()
即会返回一个StackTraceElement类的数组,该数组可以直接打印,亦可以操作。
4.3.fillInStackTrace()方法
该方法主要用于在多层捕获再抛出异常时,改变异常的堆栈轨迹,具体例子详见
Java异常的栈轨迹(Stack Trace)