我覺得 Java 的異常處理要精簡 , 因為 , 處理異常需要 JVM 付出額外的開銷 .Java 異常的根類為 java.lang.Throwable, 其有兩個直接的子類 :java.lang.Error 和 java.lang.Exception. Error 表示程序本身無法恢復的嚴重錯誤 ;Exception 類表示可以被程序捕獲并處理的 Bug. 我們也可以自定義類來拋出異常 , 但是都必須直接或間接繼承 Exception 類 .
(一) 錯誤處理與 JVM 用方法調用堆棧
MethodB()
MethodA()
Main()
MethodB()
MethodA()
Main()
如果這個時候
,
方法中的代碼可能拋出異常
,
將會有兩種處理 MethodB() MethodA() Main() MethodB() MethodA() Main()
方法 :
? 1).
? Public void methodB(int a) {
??? Try{
?????? ? //do
????? If(a<0) throw new SpecialException();
??? }catch(SpecialException e){ MethodB() MethodA() Main()
????? //do with Exception
?? }
? }
? 2). 在方法的聲明處通過 throws 語句聲明拋出異常 , 如 :
?? Public void mothodB(int a) throws SpecialException{
???? //do
???? If(a<0) throw new SpecialException();
?? }
當 java 的方法正常執行完畢 ,JVM 會從 Stack 中彈出該方法的棧結構 , 然后繼續處理前一個方法 . 如果 java 方法在執行代碼過程中拋出異常 ,JVM 必須找到能捕獲該異常的 Catch 代碼塊 . 他首先查看當前方法是否存在這樣的 Catch 代碼塊 , 如果存在 , 就執行 catch 代碼 , 否則 ,JVM 會從調用棧中彈出該方法的棧結構 , 繼續到前一個方法中查找合適的 catch 代碼塊 .
? 例如 : 當 MethodB() 方法拋出 SpecialException 時 , 如果在該方法中提供了捕獲 SpecialException 的 Catch 代碼塊 , 就執行這個異常處理代碼塊 , 否則 , 如果采用第二種方式聲明繼續拋出 SpecialException,JVM 的處理流程將退回到上層調用方法 MethodA(), 再查看 A 有沒有捕獲 SpecialException. 如果 ,A 方法中存在捕獲 SpecialException 的 catch 代碼塊 , 就執行 . 此時 A 方法定義如下 :
? Public void methodA(int a){
??? Try{
??????? methodB(a);
?????? }catch(SpecialException e){
??? ?????//do with Exception
?????? }}
如果
,A
方法也沒有捕獲
SpecialException,
而只是聲明拋出
SpecialException,JVM
處理流程將繼續退回到
main()
方法
.
如果
,main()
里面仍然沒有找到處理該異常的代碼
,
該線程就會異常終止
.
如果
,
該線程是主線程
,
應用程序也會終止
.
用戶終端就會看到原始異常信息
.
?
一般說來,在
java
程序中使用
try/catch
不會對應用程序有很大的影響,僅僅當異常發生時,
JVM
虛擬執行額外的操作,來定位處理異常的代碼塊,這時,對系統性能產生負面影響。但是,如果拋出異常的代碼和捕獲異常的代碼在一個方法中,這種影響就會小些,因此,不應該用異常處理機制來控制程序的正常流程,而應該僅僅用在程序可能出現異常的地方才使用
try/catch
。