异常是Java核心中非常重要的一个概念,但却很容易被大家忽略,这个系列我们会深入讲一下这个话题。学完这个系列,相信会让你对异常体系以及异常的各种使用场景,了如指掌,成为你职业进阶的必备技能。
名称 | 状态 |
---|---|
finally 的真正运行时机 | 已完成 |
try-with-resources 语句 | 已完成 |
Java 异常体系 | 创作中... |
catch 中的异常参数 | 创作中... |
Java 异常链 | 创作中... |
SpringMVC 中的异常处理 | 创作中... |
自定义 SpringMVC 的异常处理链 | 创作中... |
在之前的 Java语法糖 : 使用 try-with-resources 语句安全地释放资源 一文中,我们介绍了如何通过 try-with-resources 语句来代替丑陋的 finally 块。
但 try-with-resources 语句仅仅适用于自动关闭资源,有些场景下我们需要无论一段代码发生异常与否,都需要执行另外一段代码,这个时候就需要使用 finally 块了。
如果有人问你 发生异常以后 finally 块的内容什么时候会执行?你也许会毫不犹豫地回答:当然是执行完 catch 块的内容以后执行了。
但有时候你的回答也许没法这么干脆,比如下面这个例子。
当在 catch 块和 finally 块同时 return
的时候,到底会 return
什么呢?
public static int testFinally1() { try { Integer.parseInt("exception here"); } catch (Exception e) { System.out.println("catch block 1"); return 11; } finally { System.out.println("finally block 1"); return 12; } } 复制代码
事实上,即使 catch 块有 return
语句, finally 块必然执行的逻辑还是成立的。上面的方法, return
的是 12
。先输出 catch block 1
,然后输出 finally block 1
,最后返回 12
。
不知道你发现规律没有,下面再举个例子你就更清晰了。
public static int testFinally3() { try { System.out.println("start"); Integer.parseInt("testFinally3"); System.out.println("never run"); } catch (Exception e) { System.out.println("catch block 3"); return iamReturn(); } finally { System.out.println("finally block 3"); } return 31; } public static int iamReturn() { System.out.println("return block"); return 666; } 复制代码
思考一下执行这个方法会分别输出和返回什么?
先别着急看答案,根据我们之前发现的规律, finally 块必然执行,而 catch 块的非 return
部分需要在 finally 块之前执行。那么最终返回的应该是 666
。
答案确实如此。下面是控制台输出的顺序:
start catch block 3 return block finally block 3 复制代码
由此,我们可以总结 finally 块的真正运行时机了:
return
。 return
之前执行。 return
,那么 try 块和 catch 块中的 return
就没有执行机会了。 如果你真的理解了finally 块的真正运行时机,那么请思考一下,下面程序的返回值是 21
还是 22
,欢迎留言讨论。
public static int testFinally2() { try { System.out.println("start"); return 21; } catch (Exception e) { System.out.println("catch block 2"); } finally { System.out.println("finally block 2"); } return 22; } 复制代码