這篇文章帶給大家的內容是關於Java異常處理的詳細介紹,有一定的參考價值,有需要的朋友可以參考一下,希望對你有幫助。
我們來看一個例子,關於Java異常處理(Exception Handling)的一些知識點。
看下面這段程式。方法pleaseThrow接受一個Exception的實例,然後簡單地將該實例拋出。然後呼叫這個方法時,我傳入了一個SQLException的實例。因為pleaseThrow的呼叫包裹在一個try catch區塊裡,
問題:plesseThrow方法拋出的SQLException可以成功被catch住麼?
public class ExceptionForQuiz<t> { private void pleaseThrow(final Exception t) throws T { throw (T) t; } public static void main(final String[] args) { try { new ExceptionForQuiz<runtimeexception>().pleaseThrow(new SQLException()); } catch( final SQLException ex){ System.out.println("Jerry print"); ex.printStackTrace(); } } }</runtimeexception></t>
答案:上面這段程式碼有語法錯誤,不能通過編譯!
我們來一步步分析。
Java類別ExceptionForQuiz
我在實例化類別ExceptionForQuiz時,傳入的類型參數是RuntimeException。
RuntimeException在Java裡是一種Unchecked異常,即使一個方法運行時可能會拋出RuntimeException,也不需要開發人員在方法前用程式碼明確宣告。
看JDK RuntimeException的註解說的很清楚:Unchecked exceptions do NOT need to be declared in a method or constructor's clause if they can be thrown by the execution of the method or constructor.
#這個作者Frank Yellin一定是個大牛。因為泛型是Java 1.5 版本才引進的概念,關於泛型有一個類型擦除的概念,即
泛型資訊只存在於程式碼編譯階段,編譯之後的程式碼裡,與泛型相關的資訊會被擦除掉。 例如之前泛型類別中的型別參數部分如果沒有指定上限,像這種寫法
下面是從ExceptionForQuiz.class反編譯之後的程式碼:
##我們從上圖能觀察到,方法pleaseThrow和雷ExceptionForQuiz的泛型參數RuntimeException已經被擦除掉了。 pleaseThrow這個方法能拋出的異常類型已經被擦除變成Exception了。
使用javap觀察編譯產生的字節碼,同樣能發現類型參數RuntimeException被擦除的事實:
看第二個紅色高亮區域:Exceptions: throw java.lang. Exception
現在我們來看編譯器會報什麼錯誤訊息:Unreachable catch block for SQLException. This exception is never thrown from the try statement body .
根據異常類型擦除的事實,這個錯誤訊息是合理的,因為pleaseThrow方法的聲明現在只能拋出類型為Exception的異常,所以第14行的catch永遠沒有辦法接收到類型為SQLException的異常,所以編譯器拋出錯誤。
如何消除掉這個編譯器錯誤呢?把第14行的SQLException改成RuntimeException即可。
但是這樣的話,雖然消除了語法錯誤,但是方法pleaseThrow拋出的SQLException沒有辦法被catch住,會報運行時錯誤:
# 如何把pleaseThrow拋出的SQLException也用catch語句接住呢?將第14行的RuntimeException改成所有異常的超類別:Exception。 再次執行,這次既沒有語法錯誤,也沒有運行時錯誤了:SQLException已經成功地被第14行的catch語句捕捉住了。
#
以上是Java異常處理的詳細介紹的詳細內容。更多資訊請關注PHP中文網其他相關文章!