1、异常的层级结构
异常类的最顶层接口是Throwable,而其子类是Exception(异常)和Error(错误)
2、Error(错误)
Error一般是严重的错误,表示程序中无法处理的错误。
一般出现这种错误代表代码运行中JVM出现错误,比如内存不足,栈溢出等。这种错误无法捕获,将导致程序中断。我们不能去自定义Error类,也不能去捕获或者抛出Error。
3、Exception(异常)
3.1、运行时异常和非运行时异常
- 运行时异常:
编译时检查不出的异常,在程序运行中抛出的异常。程序中可以使用try catch捕获此类异常,也可以不捕获交给JVM处理。我们在程序设计时应避免这种异常。
- 非运行时异常
也叫编译异常,以除了Runtime Exception以外的异常,在编译期间就可以发现,并且不处理就会编译不通过。
3.2、受查异常和受查异常
- 受查异常
除了RuntimeException及其子类以外,其他的Exception类及其子类都属于可查异常。这种异常我们必须通过try catch处理或者通过throw或throws抛出,否则无法完成编译。
- 非受查异常
包括运行时异常(RuntimeException与其子类)和错误(Error)。
4、异常处理流程
当程序出现异常的时候:
1、JVM会在当前位置(方法)寻找异常表,寻找处理者。
2、如果找到处理者,则进行处理
3、如果没有找到合理的处理者则向上查找,也就是去该方法的调用者查找。
4、如果所有的栈帧都被弹出仍然没有处理,则抛给当前的Thread,Thread则会终止。
5、如果当前Thread为最后一个非守护线程,且未处理异常,则会导致JVM终止运行。
5、return在try catch或finally的问题
通常情况下,我们自己的业务代码,不要经常使用try catch去完成业务逻辑。因为异常机制设计的初衷是为了用于不正常的情况,所以JVM很少对其代码的性能进行优化。所以创建,抛出和捕获异常的开销是非常昂贵的。我们也尽量不要在try catch块中使用return。
但是在try和catch中使用return的代码是怎么执行的?
当try或者catch中出现throw或者return时,不会直接返回,而是先执行return中的语句,再去执行finally中的代码。
- 如果finally中没有返回,则会执行完finally后回到try或者catch中出现throw或者return的地方。执行返回
- 如果finally中有返回语句,则会直接执行finally中的返回,不会再回到try catch中。
public class TestDemo {
public static String getOut(){
try{
return "1";
}catch(Exception e){
return "2";
}finally{
return "3";
}
}
public static void main(String[] args) {
System.out.println(getOut());//3
}
}
public class TestDemo {
static int i = 0;
public static void main(String[] args) {
System.out.println(test());//结果是2
}
public static int test(){
try{
return ++i;//1、先执行++1,不会返回
}finally{
return ++i;//2、再执行finally中的++1,再返回
}
}
}
总结:当Java程序执行try块、catch块时遇到了return或throw语句,这两个语句都会导致该方法立即结束,但是系统执行这两个语句并不会结束该方法,而是去寻找该异常处理流程中是否包含finally块,如果没有finally块,程序立即执行return或throw语句,方法终止;如果有finally块,系统立即开始执行finally块。只有当finally块执行完成后,系统才会再次跳回来执行try块、catch块里的return或throw语句;如果finally块里也使用了return或throw等导致方法终止的语句,finally块已经终止了方法,系统将不会跳回去执行try块、catch块里的任何代码。