首先,需要了解异常体系的结构:
看上面的结构,Throwable是所有异常的基类,有两个子类:Error和Exception;
- Error:表示系统错误或者资源耗尽,Java系统自己使用,应用程序不该抛出和处理(但是是可以try catch的);
- Exception:表示应用程序错误;
这里需要注意,图中标出的两种异常类型:
- 受检异常(checked exception):Java强制要求必须在throws语句中抛出,调用者必须处理,否则无法编译通过;
- 未受检查异常(unchecked exception):未做要求
try-catch-finally需要注意的地方
- 异常被捕获之后,try语句内异常点之后的代码就不会再执行了,执行完了catch内的语句后,继续执行catch外的语句;
- catch方法从上至下开始匹配,找到第一个catch块后,后面的catch就不再会执行了,因此,catch的顺序需要从上至下为子类到父类;
- 同一个catch可以捕获多个异常,中间用"|"分隔;
- catch之后,可以重新抛出异常;常用于增加一些错误信息等
- finally执行的顺序要注意:如果异常发生但是没有被捕获,则finally会在异常被抛给上层之前执行;
- catch并不是必须的,可以只有try-finally;
关于finally和return语句: - try或者catch内如果有return,则return会在finally执行之后才会执行,但是finally无法改变return的值;(这里会将返回值暂时保存在一个临时变量中,
最后再返回临时变量的值) - finally中有return呢?try和catch内的return将会丢失,实际会返回finally中的return值;注意:此时,不仅会覆盖try和catch的返回值, 并且会掩盖try和catch内的异常,就像异常没有发生一样,此时异常不会再向上传递了;
- finally中如果抛出异常,原异常也会被覆盖;
- 应该尽量避免在finally中使用return或者抛出异常;
总之,不管有无return,finally都是会保证一定会执行的
throws关键字
- 用于声明一个方法可能抛出的异常
- 其调用者要不catch这个异常,或者就继续使用throws抛出
总结
- 当真正出现异常的时候,应该抛出异常,而不是返回特殊值;
- 如果知道自己如何处理异常,就进行处理;否则,就上抛;
- 上抛时,如果自己有额外信息,就应该提供这些信息,可以以原异常为cause重新抛出;