1. 程式人生 > >Java筆試面試總結—try、catch、finally語句中有return 的各類情況

Java筆試面試總結—try、catch、finally語句中有return 的各類情況

### 前言 之前在刷筆試題和麵試的時候經常會遇到或者被問到 **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 ### 有收穫?希望老鐵們來個三連擊,給更多的人看到這