在Java面试中关于finally的面试题目出现的概率非常高,而且一旦面试官问起绝不会是蜻蜓点水,而是会向你发起层层递进地“连环问”,并且回答这系列问题常常需要代码的辅助,可谓考验基础的面试利题。究竟面试官会问些什么呢?应试者又该怎样完美回答呢?今天小编就带着猿猿们亲身体验一场finally面试,希望对你有帮助。
前提需要了解:当发生异常之后,异常之后的代码不会执行,会到catch块中执行,但是catch之后的代码会执行,简单的例子:
public static String test(){ try { int i=1/0; System.out.println("error open");//上面发生异常,所以永远不会走这里 return "1"; } catch (ArithmeticException e) { System.out.println("error"); } System.out.println("try..catch...finish"); return "finish"; }
结果:
error
try..catch...finish
面试官题目
什么时候用到finally呢?finally语句在try或catch中的return语句执行之后,还是return返回之前执行呢?
考点分析
这个题目也是考查异常相关的。对于这个题目,我们通过代码来验证我们的答案,因为有异议的知识点,代码是最有说服力的。对于编程,希望你记住“不与人争辩,一切用代码说话”。
回答
什么时候用到finally呢?
某些事物(除内存外)在异常处理完后需要恢复到原始状态,如:开启的文件,网络连接等。
finally语句在try或catch中的return语句执行之后,还是return返回之前执行呢?
这个问题是一个很经典的问题,经常被面试官问,如果自己不去实验一下,可能判断就会出错。我们结合代码来分析一下。
下面我们通过4个demo来得出最终结论。
private static int test1() {
int i = 1;
try {
System.out.println("try...");
return i += 10;
} catch (Exception e) {
System.out.println("catch...");
} finally {
i++;
System.out.println("finally...");
System.out.println("i=" + i);
}
return i;
}
执行结果:
try...
finally...
i=12
test1:11
总结:finally代码块是在try代码块中的return语句执行之后,返回之前执行的。
private static int test2() {
int i = 1;
try {
System.out.println("try...");
return i += 10;
} catch (Exception e) {
System.out.println("catch...");
} finally {
i++;
System.out.println("finally...");
System.out.println("i=" + i);
return i;
}
}
执行结果:
test2:12
try...
finally...
i=12
总结:finally代码块中的return语句覆盖try代码块中的return语句。
private static Map<String, String> test3() {
Map<String, String> map = new HashMap<String, String>();
map.put("KEY", "INIT");
try {
System.out.println("try...");
map.put("KEY", "TRY");
return map;
} catch (Exception e) {
System.out.println("catch...");
map.put("KEY", "CATCH");
} finally {
System.out.println("finally...");
map.put("KEY", "FINALLY");
map = null;
}
return map;
}
执行结果:
try...
FINALLY
finally...
总结: 如果finally语句中没有return语句覆盖返回值,那么原来的返回值可能因为finally里的修改而改变也可能不变。传值类型的返回值:不变;传址类型的返回值:会变。
这里引入来一个新的问题,怎么判断一个变量是传值还是传址?传值:8种基本数据类型及其包装类,字符常量。传址:数组和对象。
private static int test4() {
int i = 1;
try {
System.out.println("try...");
i = i / 0;
return i += 10;
} catch (Exception e) {
System.out.println("catch...");
return i;
} finally {
i++;
System.out.println("finally...");
System.out.println("i=" + i);
}
}
执行结果:
try...
catch...
finally...
i=2
1
总结: try代码块中的return语句在异常的情况下不会被执行,这样具体返回哪个看情况;catch中的return执行情况与未发生异常时try中return的执行情况完全一样。
- 第五种情况:
private int test5() { int i = 1; try { System.out.println("try..."); i = i / 0; return i += 10; } catch (Exception e) { System.out.println("catch..."); return i; } finally { i++; System.out.println("finally..."); System.out.println("i=" + i); return i; } }
try...
catch...
finally...
i=2
2
总结:finally代码块中的return语句覆盖try代码块中的return语句。
汇总
根据上面的分析,咱们来汇总一下答案:
-
try语句没有被执行,如在try语句之前就返回了,这样finally语句就不会执行;因此说明了finally语句被执行的必要而非充分条件是:相应的try语句一定被执行到。
-
如果在try代码块中执行System.exit(0)语句;那么将终止Java虚拟机JVM,因此,finally语句也不会被执行到。
-
finally块的语句在try或catch中的return语句执行之后返回之前执行且finally里的修改语句可能影响也可能不影响try或catch中return已经确定的返回值,如果返回值类型为传址类型,则影响;传值类型,则不影响。若finally里也有return语句则覆盖try或catch中的return语句直接返回。