面试的时候只要涉及到笔试,几乎一定会涉及到的问题。有return的情况下try、catch、finally的问题
结论
1、finally语句块一定执行吗?
2、当try、catch中有return时,finally中的代码依然会继续执行
3、finally是在return后面的表达式运算之后执行的,此时并没有返回运算之后的值,而是把值保存起来,不管finally对该值做任何的改变,返回的值都不会改变,依然返回保存起来的值。也就是说方法的返回值是在finally运算之前就确定了的。
4、finally代码中最好不要包含return,程序会提前退出,也就是说返回的值不是try或catch中的值
举例:
情况1: try{}catch{}finally{} return;
程序正常进行
情况2: try{return;}catch{}finally{} return;
a、先执行try中的语句,包括return后面的表达式,b、然后执行finally中的语句,c、最后执行try中的return
ps:返回值是try中的return后面的表达式的值,finally后面的return语句不会被执行
情况3: try{} catch{return;} finally{} return;
先执行try中的代码块
有异常:a、先执行catch中的语句,包括return后面的表达式,b、然后执行finally中的语句,c、最后执行catch中的return,finally后面的return不会被执行
无异常:执行finally中的代码块,然后执行return语句
情况4: try{return;} catch{} finally{return;}
a、先执行try中的语句,包括return后面的表达式,b、然后执行finally中的语句,c、最后执行finally中的return
ps:返回的值是finally中return后面的表达式的值,因为finally中有return语句,所以会提前退出
情况5: try{} catch{return} finally{return};
先执行try中的代码块
有异常:a、先执行catch中的语句,包括return后面的表达式,b、然后执行finally中的语句,c、最后执行finally中的return,因为finally中有return语句,所以会提前退出
无异常:执行finally中的代码块,然后执行finally中的return
情况6: try{return;} catch{return;} finally{return;}
先执行try中的代码块,包括return后面的表达式
有异常:a、先执行catch中的语句,包括return后面的表达式,b、然后执行finally中的语句,c、最后执行finally中的return,因为finally中有return语句,所以会提前退出
无异常:执行finally中的代码块,然后执行finally中的return
最总结论:在执行try、catch中的return之前一定会执行finally中的代码(如果finally存在),如果finally中有return语句,就会直接执行finally中的return方法,所以finally中的return语句一定会被执行的。编译器把finally中的return语句标识为一个warning.
例子:
public class Test {
public int chang(){
int chang = 0;
try {
chang = 10;
return chang+=10;
}
finally{
System.out.println("finally中的结果:"+chang);
chang = 30;
}
}
public static void main(String[] args) {
Test te = new Test();
int i = te.chang();
System.out.println("main中的最后的结果:"+i);
}
}
打印的结果是:
分析结果:
1、先执行try中return后面的表达式,然后执行finally的语句
2、返回值是try中return后面的表达式计算出来的值,即使finally中对chang进行了修改,也不会修改最后的返回值。try中先把计算出来的值放到不同于chang的局部变量中,执行完finally后在取出结果。应该是保存在函数栈中的。
又找到一个例子更好的说明他们之间的执行关系
具体代码如下:
public class test {
public static String output = "";
public static void function(int i){
try{
if(i == 1){
throw new Exception();
}
output += "1";
} catch (Exception e) {
e.printStackTrace();
output += "2";
return;
}finally {
output += "3";
}
output += "4";
}
public static void main(String args[]){
function(0);
System.out.println(output); //134
function(1);
System.out.println(output); //13423
}
}
分析如下:
执行第一个function(0)时,执行try中语句,output = 1,
然后执行finally中的语句,output = 13,
然后执行finally后面的语句,output = 134,因为是static类型的变量,所以任何对其值的修改都有效。
执行第二条function(1),结果抛出异常,转到catch块,output = 1342,
finally语句在任何情况下都会执行,所以output = 13423,然后return跳出方法体,所以output=13423