大家面试的时候想必会遇到这种面试题,要我们输出结果。。。
其实在实际的业务中呢一般都是用finally来关闭流的,很少做其它业务操作,所以呢大家也就容易忽视掉这个问题
首先在finally和try中对数据的操作时数据分为 基本数据类型和引用数据类型,他们存放的地方也不一样,一个是栈区另一个是在堆区。
首先我们对基本数据类型的结果进行分析:
1 finally 中没有return
public class Main {
private static int a=2;
public static void main(String[] args) {
System.out.println("test输出的结果 "+test());
System.out.println("main 自加后 a="+(a++));
}
private static int test(){
try {
a+=3;
System.out.println("try a="+a);
return a;
}catch (Exception e){
}finally {
++a;
System.out.println("finally a="+a);
}
}
}
输出结果:
try a=5
finally a=6
test输出的结果 5
main a=6
结论: 可见方法test的返回值不受finally中的影响,但是变量a还是受到了影响,值改变了,返回值是存放在栈中的,return是就 已经把返回值压入栈了,相当于一个临时变量。在finally中并不会影响返回值
2 在finally 中加入 return
..............................
finally {
++a;
System.out.println("finally a="+a);
return a;
}
输出结果:
try a=5
finally a=6
test输出的结果 6
main a=6
结论: 此时test方法的返回值是6和a的值相等, 在try,catch中的返回值可以发现被finally的返回值给屏蔽了,所以如果finally中 有return则会屏蔽当前方法中的返回值
现在对引用变量进行分析
1 finally中没有return
public class Main {
private static List<String> list=new ArrayList<>();
public static void main(String[] args) {
System.out.println("test输出的结果 "+test());
System.out.println("main list="+list.toString());
}
private static List test(){
try {
list.add("33");
System.out.println("try list="+list.toString());
return list;
}catch (Exception e){
}finally {
list.add("66");
System.out.println("finally list="+list.toString());
}
return null;
}
}
输出结果:
try list=[33]
finally list=[33, 66]
test输出的结果 [33, 66]
main list=[33, 66]
结论: 通过结果可以知道 test的返回值内容也是跟集合list一样的, 因为list是引用类型所以堆区中存放了内存地址,在finally对值 进行改变时还是只想同一个地址,只是地址的内容变化了
2 finally中有return 、
输出值:
try list=[33]
finally list=[33, 66]
test输出的结果 [33, 66]
main list=[33, 66]
结论:可见引用类型时finally有没有return都会对当前方法的返回值产生影响
觉得文章有帮助的话就赞赏下吧!
微信:
支付宝: