前言
之前在刷筆試題和面試的時候經常會遇到或者被問到 try-catch-finally 語法塊的執行順序等問題,今天就抽空整理了一下這個知識點,然后記錄下來,
正文
本篇文章主要是通過舉例的方式來闡述各種情況,我這里根據 try-catch-finally 語法塊分為兩種大情況討論:try-catch 語法塊和 try-catch-finally 陳述句塊,然后再在每種情況里再去具體討論,
一、try-catch 陳述句塊
我們可以看看下面程式:
public static void main(String[] args) {
System.out.println(handleException0());
}
/**
* try,catch都有return
* @return
*/
private static String handleException0() {
try{
System.out.println("try開始");
String s = null;
int length = s.charAt(0);
System.out.println("try結束");
return "try塊的回傳值";
}catch (Exception e){
System.out.println("捕獲到了例外");
return "catch的回傳值";
}
}
執行結果:
try開始
捕獲到了例外
catch的回傳值
分析:程式首先執行 try 塊里面的代碼,try 塊里面發現有例外,try 塊后面的代碼不會執行(自然也不會return),然后進入匹配例外的那個 catch 塊,然后進入 catch 塊里面將代碼執行完畢,當執行到 catch 里面的return 陳述句的時候,程式中止,然后將此 return 的最終結果回傳回去,
二、try-catch-finally 陳述句塊
這種語法塊我分為了 4 種情況討論,下面進行一一列舉,
1、第一種情況,try 塊里面有 return 的情況,并且捕獲到例外
例1:
public static void main(String[] args) {
String result = handleException1();
System.out.println(result);
}
private static String handleException1() {
try{
System.out.println("try開始");
String str = null;
int length = str.length();
System.out.println("try結束");
}catch (Exception e){
System.out.println("捕獲到了例外");
}finally {
System.out.println("finally塊執行完畢了");
}
return "最終的結果";
}
例1執行的結果如下:
try開始
捕獲到了例外
finally塊執行完畢了
最終的結果
例2:
public static void main(String[] args) {
String result = handleException2();
System.out.println(result);
}
private static String handleException2() {
try{
System.out.println("try開始");
String str = null;
int length = str.length();
System.out.println("try結束");
return "try塊的回傳值";
}catch (Exception e){
System.out.println("捕獲到了例外");
}finally {
System.out.println("finally塊執行完畢了");
}
return "最終的結果";
}
例2的執行結果如下:
try開始
捕獲到了例外
finally塊執行完畢了
最終的結果
分析:首先 例1 和 例2 的結果是很顯然的,當遇到例外的時候,直接進入匹配到相對應的 catch 塊,然后繼續執行 finallly 陳述句塊,最后將 return 結果回傳回去,
第二種情況:try塊里面有return的情況,但是不會捕獲到例外
例3:
思考:下面代碼try陳述句塊中有return陳述句,那么是否執行完try陳述句塊就直接return退出方法了呢?
public static void main(String[] args) {
String result = handleException3();
System.out.println(result);
}
private static String handleException3() {
try{
System.out.println("");
return "try塊的回傳值";
}catch (Exception e){
System.out.println("捕獲到了例外");
}finally {
System.out.println("finally塊執行完畢了");
}
return "最終的結果";
}
例3的執行結果如下:
finally塊執行完畢了
try塊的回傳值
分析:例3的結果其實我們可以通過打斷點的方式去看看程式的具體執行流程,通過打斷點我們可以發現,代碼先執行 try塊 里的代碼,當執行到 return 陳述句的時候,handleException3方法并沒有立刻結束,而是繼續執行finally塊里的代碼,finally塊里的代碼執行完后,緊接著回到 try 塊的 return 陳述句,再把最終結果回傳回去, handleException 方法執行完畢,
第三種情況:try塊和finally里面都有return的情況
例4:
public static void main(String[] args) {
System.out.println(handleException4());
}
/**
* 情況3:try和finally中均有return
* @return
*/
private static String handleException4() {
try{
System.out.println("");
return "try塊的回傳值";
}catch (Exception e){
System.out.println("捕獲到了例外");
}finally {
System.out.println("finally塊執行完畢了");
return "finally的回傳值";
}
// return "最終的結果";//不能再有回傳值
}
例4的執行結果:
finally塊執行完畢了
finally的回傳值
分析:需要注意的是,當 try 塊和 finally 里面都有 return 的時候,在 try/catch/finally 語法塊之外不允許再有return 關鍵字,我們還是通過在程式中打斷點的方式來看看代碼的具體執行流程,代碼首先執行 try 塊 里的代碼,當執行到 return 陳述句的時候,handleException4 方法并沒有立刻結束,而是繼續執行 finally 塊里的代碼,當發現 finally 塊里有 return 的時候,直接將 finally 里的回傳值(也就是最終結果)回傳回去, handleException4 方法執行完畢,
第四種情況:try塊,catch塊,finally塊都有return
例5:
public static void main(String[] args) {
System.out.println(handleException5());
}
/**
* 情況4:try,catch,finally都有return
* @return
*/
private static String handleException5() {
try{
System.out.println("try開始");
int[] array = {1, 2, 3};
int i = array[10];
System.out.println("try結束");
return "try塊的回傳值";
}catch (Exception e){
e.printStackTrace();//這行代碼其實就是列印輸出例外的具體資訊
System.out.println("捕獲到了例外");
return "catch的回傳值";
}finally {
System.out.println("finally塊執行完畢了");
return "finally的回傳值";
}
// return "最終的結果";
}
例5的執行結果:
try開始
捕獲到了例外
finally塊執行完畢了
finally的回傳值
java.lang.ArrayIndexOutOfBoundsException: 10
at com.example.javabasic.javabasic.ExceptionAndError.TryCatchFinally.handleException5(TryCatchFinally.java:25)
at com.example.javabasic.javabasic.ExceptionAndError.TryCatchFinally.main(TryCatchFinally.java:14)
分析:程式首先執行try塊里面的代碼,try塊里面發現有例外,try塊后面的代碼不會執行(自然也不會return),然后進入匹配例外的那個catch塊,然后進入catch塊里面將代碼執行完畢,當執行到catch里面的return陳述句的時候,程式不會馬上終止,而是繼續執行finally塊的代碼,最后執行finally里面的return,然后將此return的最終結果回傳回去,
總結
其實,我們通過以上例子我們可以發現,不管return關鍵字在哪,finally一定會執行完畢,理論上來說try、catch、finally塊中都允許書寫return關鍵字,但是執行優先級較低的塊中的return關鍵字定義的回傳值將覆寫執行優先級較高的塊中return關鍵字定義的回傳值,也就是說finally塊中定義的回傳值將會覆寫catch塊、try塊中定義的回傳值;catch塊中定義的回傳值將會覆寫try塊中定義的回傳值,
再換句話說如果在finally塊中通過return關鍵字定義了回傳值,那么之前所有通過return關鍵字定義的回傳值都將失效——因為finally塊中的代碼一定是會執行的,
公眾號:良許Linux
有識訓?希望老鐵們來個三連擊,給更多的人看到這篇文章
轉載請註明出處,本文鏈接:https://www.uj5u.com/caozuo/47074.html
標籤:Linux
上一篇:Java工具類—包裝類
