昨天的面試官居然是位小姐姐,聊了半個多小時的異常處理


大風吹去了往日的霧霾,陽光透過窗戶照進來,透過窗戶可以看到遠處的山脈與藍天相接,這可比我那永遠見不到陽光的出租屋好多了。漸漸走進的腳步聲打斷了我的思緒,一位小姐姐坐在了面前,甜甜的香水味立刻鑽進了我的鼻孔。

小姐姐微笑地說:”您好,我是今天的面試官,那么開始吧?“

我收起直勾勾的眼睛,說:“好的。”

小姐姐說:“在Java的異常處理中有兩大組成要素:拋出異常和捕獲異常。那么拋出異常可以分為哪兩種呢?”

我立刻回答到:

拋出異常可以分為顯式和隱式。顯式拋異常是在代碼中使用throw關鍵字手動將異常實例拋出。隱式拋異常是 Java 虛擬機在執行過程中,遇到無法繼續執行的異常狀態,自動將異常實例拋出,比如我們經常遇到的空指針異常(NullPointerException)。

小姐姐說:“很好,那么捕獲異常中經常用到哪些關鍵字呢?”

我立刻回答到:

我們一般用到trycatchfinally等關鍵字。try被用來標記需要進行異常監控的代碼;catch被用來捕獲在try 監控的代碼中觸發的某種指定類型的異常,還可以定義針對該異常類型進行如何處理;finally被用來聲明一段無論發生什么異常都必定運行的代碼,它避免跳過某些關鍵的清理代碼,比如:關閉已打開的IO資源。

小姐姐說:“很好,如果三個關鍵字一起使用,代碼執行的順序是什么樣子的?”

我立刻回答到:

在正常執行的情況下,先執行try中的代碼再執行finally中的代碼。

如果try中的代碼觸發異常,並且異常沒有被捕獲,finally中代碼會被直接執行,並且在執行之后重新拋出該異常;如果異常被catch捕獲,先執行catch中的代碼再執行finally中的代碼。

如果catch中的代碼也觸發了異常,那么finally中代碼同樣會被執行,並會拋出catch代碼觸發的那個異常。如果finally中的代碼也觸發了異常,那么會中斷當前finally 代碼的執行,並拋出異常。

小姐姐說:“很好,在Java虛擬機中,是通過什么方式實現異常處理?”

這個問題有點難度,我稍微思考了一下回答到:

主要是通過異常表。在編譯生成的字節碼中,每個方法都附帶一個異常表。異常表中可能有多條記錄,每一條記錄都包括from指針、to 指針、target指針和所捕獲的異常類型。這些指針的值是字節碼索引(bytecode index),用於定位字節碼。

其中,from指針和to指針表示異常處理監控的范圍,比如: try所覆蓋的范圍。target指針指向異常處理代碼的起始位置,比如:catch代碼的起始位置。

如果有異常觸發時,Java虛擬機會從上至下遍歷異常表中的所有記錄。當觸發異常的字節碼的索引值在某個異常表記錄的監控范圍內,Java虛擬機會判斷所拋出的異常和該記錄想要捕獲的異常是否匹配。如果匹配,控制流轉將會移至該記錄 target 指針指向的字節碼。

小姐姐說:“有點抽象,可以舉個例子嗎?”

“當然可以。”我一邊說,一邊在紙上寫了起來:

public class OneMore {

    public static void main(String[] args) {
        String str = "萬貓學社";
        try {
            str = "try";
        } catch (Exception e) {
            str = "catch";
        }
    }
}

這段代碼的 main 方法中,我定義了一段trycatch 代碼。編譯過后的字節碼中,這個方法的異常表擁有一個記錄:

public static void main(java.lang.String[]);
  descriptor: ([Ljava/lang/String;)V
  flags: ACC_PUBLIC, ACC_STATIC
  Code:
    stack=1, locals=3, args_size=1
       0: ldc           #2                  // String 萬貓學社
       2: astore_1
       3: ldc           #3                  // String try
       5: astore_1
       6: goto          13
       9: astore_2
      10: ldc           #5                  // String catch
      12: astore_1
      13: return
    Exception table:
       from    to  target type
           3     6     9   Class java/lang/Exception

from指針和to指針分別為 3 和 6,代表它的監控范圍從索引為 3 的字節碼開始,到索引為 6 的字節碼結束(不包括 6)。該記錄的 target 指針是 9,代表這個異常處理從索引為 9 的字節碼開始。該記錄的最后一列,代表該異常處理所捕獲的異常類型是Exception

當觸發異常的字節碼的索引值在 3 和 6 之間時,Java虛擬機會判斷所拋出的異常是否時Exception。如果是,控制流轉將會移至索引為 9 的字節碼開始執行。

小姐姐滿意地說:“很好,我去叫HR和你聊聊,如果順利的話,今天就可以發offer。”


參考文獻:
《Java編程思想》
《Java核心技術》
《深入理解Java虛擬機:JVM高級特性與實踐》


竟然已經看到這里了,你我定是有緣人,留下你的點贊關注,他日必成大器。

微信公眾號:萬貓學社

微信掃描二維碼

關注后回復「電子書」

獲取12本Java必讀技術書籍


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM