1. 程式人生 > >【Java入門提高篇】Day16 Java異常處理(下)

【Java入門提高篇】Day16 Java異常處理(下)

http final 自定義異常 2個 輸入 也有 推薦 錯誤 num

  今天繼續講解java中的異常處理機制,主要介紹Exception家族的主要成員,自定義異常,以及異常處理的正確姿勢。

Exception家族

  一圖勝千言,先來看一張圖。

技術分享圖片

  Exception這是一個父類,它有兩個兒子,IOException和RuntimeException,每個兒子都很能生,所以它有著一堆的孫子,但其實,Exception家族還有一個大家夥,那就是Throwable,這是一個接口,看名字就知道意思,就是“可被拋出”嘛,它還有一個同父異母的哥哥,那就是Error,這家夥可厲害了,Error類一般是指與虛擬機相關的問題,如系統崩潰,虛擬機錯誤,內存空間不足,方法調用棧溢出等。catch語句裏,不僅可以catch住Exception,還能catch住Error(什麽?你真的打算catch Error??程獨秀同學,你先坐下。)一般情況下,是不能捕獲Error的,對於這類錯誤,Java編譯器不去檢查他們。對於這類錯誤的導致的應用程序中斷,僅靠程序本身無法恢復和預防,遇到這樣的錯誤,建議讓程序終止。除非你有把握能正確處理,否則程獨秀同學還是坐下吧(滑稽)。

Unchecked Exception和Checked Exception

  你也許會一臉懵逼,???,這是啥?異常也是分派別的,Unchecked Exception表示“未檢查異常“,Checked Exception自然就是”已檢查異常“,派生於Error或者RuntimeException的異常稱為unchecked異常,所有其他的異常成為checked異常。那問題來了,為啥要區分這兩種異常?

  我們可以再看看上面那個圖,可以看出,RuntimeException和Error都是由程序內部引發的錯誤,比如上一篇裏所說的空指針和算術異常。而Checked Exception則大都是由外部因素導致的,如文件無法找到異常,這是虛擬機無法掌控的情況,當出現異常,虛擬機也只能一臉懵逼,不知道該如何是好,所以當有可能發生時,就必須要使用try..catch去捕獲它,而對於Unchecked Exception 時,大部分是由於代碼引發的,所以只要代碼寫的足夠完善,是不會拋出這樣的異常的,所以也不強制要求捕獲。

  所以原因其實很簡單,編譯器將檢查你是否為所有的已檢查異常提供了異常處理機制,比如說我們使用Class.forName()來查找給定的字符串的class對象的時候,如果沒有為這個方法提供異常處理,編譯是無法通過的。已檢查異常的意義就在於讓你知道,這地方是有可能拋異常的,你要註意了,趕緊捕獲了。

自定義異常

  那麽如何自定義一個異常呢?其實很簡單,只需要繼承Exception類就好了。看下面的栗子:

public class MyException extends Exception {
    public MyException() {
        super();
    }

    
public MyException(String message) { super(message); } public MyException(String message, Throwable cause) { super(message, cause); } public MyException(Throwable cause) { super(cause); } protected MyException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) { super(message, cause, enableSuppression, writableStackTrace); } }

  MyException繼承了Exception類,重寫了構造函數,並沒有加自己的邏輯,只是調用了父類的方法。你看,自定義一個異常其實很簡單吧。看到這你也許又疑惑了,這尼瑪好像就是給Exception換了個名字,有啥用???

  別急,別急,你忘了嗎,Exception不僅是可以捕獲的,還是可以主動拋出的,所以當遇到某些特定的情況時,我們就可以主動拋出異常,然後在調用時去捕獲它,獲取異常信息,如果直接用Exception的話,那麽捕獲的時候,會把所有的異常,該捕獲不該捕獲的都一起捕獲了,那麽就沒法區分哪些是我們主動拋出來的異常了,這樣就無法對那些異常進行特殊處理了。

異常處理的正確姿勢  

  接下來要簡單介紹一個實際使用中常用的異常處理方法——異常鏈化處理。

  在一些大型的,模塊化的軟件開發中,一旦一個地方發生異常,則如骨牌效應一樣,將導致出現一連串的異常。假設B模塊需要調用A模塊的方法,如果A模塊發生異常,則B也將不能完成而發生異常,但是B在拋出異常時,會將A的異常信息掩蓋掉,這將使得異常的根源信息丟失。而使用異常的鏈化可以將多個模塊的異常串聯起來,使得異常信息不會丟失。

  異常鏈化就是用一個異常對象為參數構造新的異常對象。新的異對象將包含先前異常的信息。這項技術主要是異常類的一個帶Throwable參數的函數來實現的。這個當做參數的異常,我們叫他根源異常(cause)。如果你細心一點的話,會發現上面的栗子裏也有一個叫做cause的東西,沒錯,說的其實就是它,在new一個新的異常時,將之前的異常信息傳入構造函數即可。下面再用一個簡單的栗子進行說明:

public class Test {
    public static void main(String[] args) {
        System.out.println("請輸入2個加數");
        int result;
        try {
            result = add();
            System.out.println("結果:"+result);
        } catch (Exception e){
            e.printStackTrace();
        }
    }

    /**
     * 執行加法計算
     */
    private static int add() throws Exception {
        int result;
        try {
            List<Integer> nums =getInputNumbers();
            result = nums.get(0)  + nums.get(1);
        }catch(InputMismatchException immExp){
            //鏈化:以一個異常對象為參數構造新的異常對象。
            throw new Exception("計算失敗",immExp);
        }
        return  result;
    }

    /**
     * 獲取輸入的整數
     */
    private static List<Integer> getInputNumbers() {
        List<Integer> nums = new ArrayList<>();
        Scanner scan = new Scanner(System.in);
        try {
            int num1 = scan.nextInt();
            int num2 = scan.nextInt();
            nums.add(new Integer(num1));
            nums.add(new Integer(num2));
        }catch(InputMismatchException immExp){
            throw immExp;
        }finally {
            scan.close();
        }
        return nums;
    }
}

  輸出如下:

請輸入2個加數
d d
java.lang.Exception: 計算失敗
    at com.frank.chapter17.Test.add(Test.java:35)
    at com.frank.chapter17.Test.main(Test.java:18)
Caused by: java.util.InputMismatchException
    at java.util.Scanner.throwFor(Scanner.java:864)
    at java.util.Scanner.next(Scanner.java:1485)
    at java.util.Scanner.nextInt(Scanner.java:2117)
    at java.util.Scanner.nextInt(Scanner.java:2076)
    at com.frank.chapter17.Test.getInputNumbers(Test.java:47)
    at com.frank.chapter17.Test.add(Test.java:31)
    ... 1 more

  可以看到,當輸入的不是整數時,發生了異常,在getInputNumbers方法裏沒有處理這個異常,而是將它繼續拋出,在add方法裏捕獲了異常之後,以該異常為構造參數,重新拋出了一個異常,從打印輸出的信息可以看到,不僅僅有第二次拋出的異常信息,第一次的輸出信息不匹配異常的詳細信息也包含在了裏面,銜接在Caused by之後,形成了一條異常鏈,這樣可以方便我們更快的排查問題所在。

  至此,異常就講解完畢了,希望能給大家帶來一些啟發和思考,如果覺得還算ok的話,記得動動小手點推薦,讓更多人可以看到,也歡迎關註我的博客,會持續更新的。如果有什麽講的不好的地方。。。emmmmmm,你倒是來打我呀(逃)

【Java入門提高篇】Day16 Java異常處理(下)