1. 程式人生 > >Java中的Error與Exception

Java中的Error與Exception

Error和Exception的聯絡

  • 繼承結構:Error和Exception都是繼承於Throwable,RuntimeException繼承自Exception。

  • Error和RuntimeException及其子類稱為未檢查異常(Unchecked exception),其它異常成為受檢查異常(Checked Exception)。

Error和Exception的區別

  • Error類一般是指與虛擬機器相關的問題,如系統崩潰,虛擬機器錯誤,記憶體空間不足,方法呼叫棧溢位等。如java.lang.StackOverFlowError和Java.lang.OutOfMemoryError。對於這類錯誤,Java編譯器不去檢查他們。對於這類錯誤的導致的應用程式中斷,僅靠程式本身無法恢復和預防,遇到這樣的錯誤,建議讓程式終止。

  • Exception類表示程式可以處理的異常,可以捕獲且可能恢復。遇到這類異常,應該儘可能處理異常,使程式恢復執行,而不應該隨意終止異常。

執行時異常和受檢查的異常

Exception又分為執行時異常(Runtime Exception)和受檢查的異常(Checked Exception )。

  • RuntimeException:其特點是Java編譯器不去檢查它,也就是說,當程式中可能出現這類異常時,即使沒有用try……catch捕獲,也沒有用throws丟擲,還是會編譯通過,如除數為零的ArithmeticException、錯誤的型別轉換、陣列越界訪問和試圖訪問空指標等。處理RuntimeException的原則是:如果出現RuntimeException,那麼一定是程式設計師的錯誤。

  • 受檢查的異常(IOException等):這類異常如果沒有try……catch也沒有throws丟擲,編譯是通不過的。這類異常一般是外部錯誤,例如檔案找不到、試圖從檔案尾後讀取資料等,這並不是程式本身的錯誤,而是在應用環境中出現的外部錯誤。

throw 和 throws兩個關鍵字有什麼不同

  • throw 是用來丟擲任意異常的,你可以丟擲任意 Throwable,包括自定義的異常類物件;throws總是出現在一個函式頭中,用來標明該成員函式可能丟擲的各種異常。如果方法丟擲了異常,那麼呼叫這個方法的時候就需要處理這個異常。

try-catch-finally-return執行順序

  • 1、不管是否有異常產生,finally塊中程式碼都會執行;
  • 2、當try和catch中有return語句時,finally塊仍然會執行;
  • 3、finally是在return後面的表示式運算後執行的,所以函式返回值是在finally執行前確定的。無論finally中的程式碼怎麼樣,返回的值都不會改變,仍然是之前return語句中儲存的值;
  • 4、finally中最好不要包含return,否則程式會提前退出,返回值不是try或catch中儲存的返回值。

舉例:

情況1:try{} catch(){}finally{} return;
按正常順序執行。

情況2:try{ return; }catch(){} finally{} return;
程式執行try塊中return之前(包括return語句中的表示式運算)程式碼;
再執行finally塊,最後執行try中return;
finally塊後面的return語句不再執行。

情況3:try{ } catch(){return;} finally{} return;
程式先執行try,如果遇到異常執行catch塊,
有異常:
則執行catch中return之前(包括return語句中的表示式運算)程式碼,再執行finally語句中全部程式碼,
最後執行catch塊中return. finally塊後面的return語句不再執行。
無異常:
執行完try再finally再執行最後的return語句.

情況4:try{ return; }catch(){} finally{return;}
程式執行try塊中return之前(包括return語句中的表示式運算)程式碼;
再執行finally塊,因為finally塊中有return所以提前退出。

情況5:try{} catch(){return;}finally{return;}
程式執行catch塊中return之前(包括return語句中的表示式運算)程式碼;
再執行finally塊,因為finally塊中有return所以提前退出。

情況6:try{ return;}catch(){return;} finally{return;}
程式執行try塊中return之前(包括return語句中的表示式運算)程式碼;
有異常:執行catch塊中return之前(包括return語句中的表示式運算)程式碼;
則再執行finally塊,因為finally塊中有return所以提前退出。
無異常:則再執行finally塊,因為finally塊中有return所以提前退出。

測試程式事例

public class FinallyTest  
{
    public static void main(String[] args) {
        System.out.println(new FinallyTest().test());;
    }
    static int test()
    {
        int x = 1;
        try
        {
            x++;
            return x;
        }
        finally
        {
            ++x;
        }
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20

列印結果是2。

根據之前的分析,在try語句塊中,在執行return語句時,要返回的結果已經準備好了,然後程式轉到finally執行。在轉去之前,try中先把要返回的結果存放到不同於x的區域性變數中去,執行完finally之後,再從中取出返回結果,因此,即使finally中對變數x進行了改變,也不會影響返回結果。