大風吹去了往日的霧霾,陽光透過窗戶照進來,透過窗戶可以看到遠處的山脈與藍天相接,這可比我那永遠見不到陽光的出租屋好多了。漸漸走進的腳步聲打斷了我的思緒,一位小姐姐坐在了面前,甜甜的香水味立刻鑽進了我的鼻孔。
小姐姐微笑地說:”您好,我是今天的面試官,那么開始吧?“
我收起直勾勾的眼睛,說:“好的。”
小姐姐說:“在Java的異常處理中有兩大組成要素:拋出異常和捕獲異常。那么拋出異常可以分為哪兩種呢?”
我立刻回答到:
拋出異常可以分為顯式和隱式。顯式拋異常是在代碼中使用throw
關鍵字手動將異常實例拋出。隱式拋異常是 Java 虛擬機在執行過程中,遇到無法繼續執行的異常狀態,自動將異常實例拋出,比如我們經常遇到的空指針異常(NullPointerException)。
小姐姐說:“很好,那么捕獲異常中經常用到哪些關鍵字呢?”
我立刻回答到:
我們一般用到try
、catch
、finally
等關鍵字。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 方法中,我定義了一段try
和catch
代碼。編譯過后的字節碼中,這個方法的異常表擁有一個記錄:
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必讀技術書籍
