如题,在很多笔试或者面试中,会提及到try-catch-finally的返回值问题或者执行顺序问题,下面用一个简单的例子进行测试。
结论:
一、执行顺序
- 执行try块;
- 如果try块中有异常,则执行catch块;
- 无论有无异常,都会执行finally块;
- 但是要注意,如果在try块或者catch块中添加了System.exit(0),程序会退出,finally不会执行。
System.exit(0):将停止当前线程和所有其他当场死亡的线程。finally块并不能让已经停止的线程继续执行。
当System.exit(0)被调用时,虚拟机退出当前要执行两项清理工作:
(1)执行系统中注册的所有关闭钩子;
(2)如果程序调用了System.runFinalizerOnExit(true);,那么JVM会对所有的还未结束的对象调用Finalizer。
二、返回值问题-return
- 无论try-catch中有无异常,如果finally块中有return语句,最后返回的是finally的return值;
- 如果finally中没有return语句,若try中有异常,则返回catch中的return值,反之,则返回try中return值。在这种情况下,return语句是在finally之后执行的。
代码分析如下:
package com.trycatch;
public class Test {
public static void main(String[] args) {
System.out.println("返回值:"+returnTest());
}
private static int returnTest(){
int reval = -2;
try{
System.out.println("try块中:"+reval);
reval = 1;
return reval;
}catch(Exception e){
//System.exit(0);
System.out.println("catch块中:"+reval);
reval = -1;
return reval;
}finally{
System.out.println("finally块中:"+reval);
reval = 0;
return reval;
}
}
}
程序无异常,代码中返回值reval进过了try的赋值,最后到了finally中赋值为0,所以结果返回0;
try块中:-2
finally块中:1
返回值:0
若在try中添加一个异常int a = 1 / 0;则会先执行catch中的赋值return,最后也要执行finally中的返回值。
续加:try-finally形式的模块,在平时很少用,这样用有一个很明显的目的就是如果出现异常,则当前程序不处理,只需要在finally清理资源并将该无法处理的异常抛给它的用户自行处理,这样实现了异常处理(throw)与异常清理(finally)的解耦。