Java中的異常 Exception

  java.lang.Exception類是Java中全部異常的直接或間接父類。即Exception類是全部異常的根類。java

  好比程序: 面試

複製代碼

public class ExceptionTest
{
      public static void main(String[] args)
      {
             int a = 3;
             int b = 0;
             int c = a / b;          
             System.out.println(c);
      }
}

複製代碼

 

  編譯經過,執行時結果:spa

  Exception in thread "main" java.lang.ArithmeticException: / by zero指針

     at com.learnjava.exception.ExceptionTest.main(ExceptionTest.java:9)繼承

  由於除數爲0,因此引起了算數異常。開發

 

  比較常見的異常還有這種:空指針異常字符串

  java.lang.NullPointerException是空指針異常,出現該異常的緣由在於某個引用爲null,但卻調用了它的某個方法,這時就會出現該異常。編譯器

 

Java中的異常分爲兩大類:

  1.Checked ExceptionRuntime Exception虛擬機

  2.Unchecked ExceptionRuntime Exceptionit

運行時異常

  RuntimeException類是Exception類的子類,它叫作運行時異常,Java中的全部運行時異常都會直接或者間接地繼承自RuntimeException類。

  Java中凡是繼承自Exception,而不繼承自RuntimeException類的異常都是非運行時異常

 

異常處理的通常結構

複製代碼

try
    {
         // 可能發生異常的代碼
        // 若是發生了異常,那麼異常以後的代碼都不會被執行
    }
    catch (Exception e)
    {
        // 異常處理代碼
    }
    finally
    {
        // 無論有沒有發生異常,finally語句塊都會被執行
    }

複製代碼

  好比本文最開始的除法運算代碼,加入異常處理以後: 

複製代碼

public class ExceptionTest
{
    public static void main(String[] args)
    {
        int c = 0;
        try
        {
            int a = 3;
            int b = 0;

            // 這塊代碼出現了異常
            c = a / b;

            // 那麼異常以後的代碼都不會被執行
            System.out.println("Hello World");
        }
        catch (ArithmeticException e)
        {
            e.printStackTrace();
        }
        finally
        {
            //無論有沒有發生異常,finally語句塊都會被執行
            System.out.println("Welcome");
        }

        System.out.println(c);
        // 當b爲0時,有異常,輸出爲c的初始值0
    }
}

複製代碼

 

多個catch

  一個try後面能夠跟多個catch,但無論多少個,最多隻會有一個catch塊被執行。

 

異常處理方法

  對於非運行時異常(checked exception),必需要對其進行處理,不然沒法經過編譯。

  處理方式有兩種:

  1.使用try..catch..finally進行捕獲;

  2.在產生異常的方法聲明後面寫上throws 某一個Exception類型,如throws Exception,將異常拋出到外面一層去。

  對非運行時異常的處理詳見代碼例子:

  處理方式1:將異常捕獲

複製代碼

public class ExceptionTest2
{
    public void method() throws Exception // 將異常拋出,由調用這個方法的方法去處理這個異常,若是main方法也將異常拋出,則交給Java虛擬機來處理
    {
        System.out.println("Hello World");

        // 拋出異常
        throw new Exception();
    }

    public static void main(String[] args)
    {
        ExceptionTest2 test = new ExceptionTest2();

        try
        {
            test.method();
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        finally
        {
            System.out.println("Welcome");
        }


    }

}

複製代碼

  處理方式2:將異常繼續向外拋出

複製代碼

public class ExceptionTest2
{
    public void method() throws Exception // 將異常拋出,由調用這個方法的方法去處理這個異常,若是main方法也將異常拋出,則交給Java虛擬機來處理
    {
        System.out.println("Hello World");

        // 拋出異常
        throw new Exception();
    }

    public static void main(String[] args) throws Exception // main方法選擇將異常繼續拋出
    {
        ExceptionTest2 test = new ExceptionTest2();

        test.method(); // main方法須要對異常進行處理

        // 執行結果:
        // Hello World
        // Exception in thread "main" java.lang.Exception
        // at com.learnjava.exception.ExceptionTest2.method(ExceptionTest2.java:10)
        // at com.learnjava.exception.ExceptionTest2.main(ExceptionTest2.java:17)
    }

}

複製代碼

 

  對於運行時異常(runtime exception),能夠對其進行處理,也能夠不處理。推薦不對運行時異常進行處理。

 

自定義異常

  所謂自定義異常,一般就是定義一個類,去繼承Exception類或者它的子類。由於異常必須直接或者間接地繼承自Exception類。

  一般狀況下,會直接繼承自Exception類,通常不會繼承某個運行時的異常類。

  自定義異常能夠用於處理用戶登陸錯誤,用戶輸入錯誤提示等。

  自定義異常的例子:

  自定義一個異常類型: 

複製代碼

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

複製代碼

  一種異常處理方式:

複製代碼

public class ExceptionTest4
{

    public void method(String str) throws MyException
    {
        if(null == str)
        {
            throw new MyException("傳入的字符串參數不能爲null!");
        }
        else
        {
            System.out.println(str);
        }
    }
    
    public static void main(String[] args) throws MyException //異常處理方式1,不斷向外拋出
    {
        ExceptionTest4 test = new ExceptionTest4();
        test.method(null);
    }
}

複製代碼

  另外一種異常處理方式:

複製代碼

public class ExceptionTest4
{

    public void method(String str) throws MyException
    {
        if (null == str)
        {
            throw new MyException("傳入的字符串參數不能爲null!");
        }
        else
        {
            System.out.println(str);
        }
    }

    public static void main(String[] args)
    {
        //異常處理方式2,採用try...catch語句
        try
        {
            ExceptionTest4 test = new ExceptionTest4();
            test.method(null);

        }
        catch (MyException e)
        {
            e.printStackTrace();
        }    
        finally
        {
            System.out.println("程序處理完畢");
        }

    }
}

複製代碼

 

  前面說過,能夠有多個catch塊,去捕獲不一樣的異常,真正執行的時候最多隻進入一個catch塊

  下面這個例子,定義了兩種自定義的異常類型:

複製代碼

public class MyException extends Exception
{

    public MyException()
    {
        super();
    }
    
    public MyException(String message)
    {
        super(message);
    }
}


public class MyException2 extends Exception
{
    public MyException2()
    {
        super();
    }
    public MyException2(String message)
    {
        super(message);
    }

}


public class ExceptionTest4
{

    public void method(String str) throws MyException, MyException2
    {
        if (null == str)
        {
            throw new MyException("傳入的字符串參數不能爲null!");
        }
        else if ("hello".equals(str))
        {
            throw new MyException2("傳入的字符串不能爲hello");
        }
        else
        {
            System.out.println(str);
        }
    }

    public static void main(String[] args)
    {
        // 異常處理方式2,採用try...catch語句
        try
        {
            ExceptionTest4 test = new ExceptionTest4();
            test.method(null);

        }
        catch (MyException e)
        {
            System.out.println("進入到MyException catch塊");
            e.printStackTrace();
        }
        catch (MyException2 e)
        {
            System.out.println("進入到MyException2 catch塊");
            e.printStackTrace();
        }
        finally
        {
            System.out.println("程序處理完畢");
        }

    }
}

複製代碼

 

  咱們可使用多個catch塊來捕獲異常,這時須要將父類型的catch塊放到子類型的catch塊以後,這樣才能保證後續的catch塊可能被執行,不然子類型的catch塊將永遠沒法到達,Java編譯器會報錯。

  若是異常類型是獨立的,那麼它們的先後順序沒有要求。

  如對上面的代碼進行改動後,以下列出:

複製代碼

public class ExceptionTest4
{

    public void method(String str) throws Exception // 也能夠聲明Exception,只要聲明的能夠涵蓋全部拋出的異常便可
    {
        if (null == str)
        {
            throw new MyException("傳入的字符串參數不能爲null!");
        }
        else if ("hello".equals(str))
        {
            throw new MyException2("傳入的字符串不能爲hello");
        }
        else
        {
            System.out.println(str);
        }
    }

    public static void main(String[] args)
    {
        // 異常處理方式2,採用try...catch語句
        try
        {
            ExceptionTest4 test = new ExceptionTest4();
            test.method(null);

        }
        catch (MyException e)
        {
            System.out.println("進入到MyException catch塊");
            e.printStackTrace();
        }
        catch (MyException2 e)
        {
            System.out.println("進入到MyException2 catch塊");
            e.printStackTrace();
        }
        catch (Exception e)
        {
            //雖然須要加上,可是這塊代碼不會被執行,只是爲了編譯成功
            System.out.println("進入到MyException catch塊");
            e.printStackTrace();
            //若是去掉前面兩個catch塊或其中之一,則發生該異常時就會進入此catch塊
            //catch塊的匹配是按照從上到下的順序,因此這個塊若是放在最前面就會捕獲全部的異常,後面的塊永遠不會執行,這時候會提示編譯錯誤
        }
        finally
        {
            System.out.println("程序處理完畢");
        }

    }
}

複製代碼

 

面試常考題型

  try塊中的退出語句

  雖然實際開發中不會遇到這樣的狀況,可是筆試面試時有關異常常常會問到以下狀況:

複製代碼

public class ExceptionTest5
{

    public void method()
    {
        try
        {
            System.out.println("進入到try塊");
            
            //return;
            //會先執行finally塊再返回
            
            //虛擬機退出
            //System.exit(0);
            //不會執行finally塊中的語句,直接退出
        }
        catch (Exception e)
        {
            System.out.println("異常發生了!");
            
        }
        finally
        {
            System.out.println("進入到finally塊");
            
        }
        
        System.out.println("後續代碼");
        
    }
    
    public static void main(String[] args)
    {
        ExceptionTest5 test = new ExceptionTest5();
        test.method();
    }
}

複製代碼


  在加上return語句前,程序輸出:

    進入到try塊

    進入到finally塊

    後續代碼

 

  若是在try塊中加入return語句:

  程序執行輸出:

    進入到try塊

    進入到finally塊

 

  說明try塊中有return語句時,仍然會首先執行finally塊中的語句,而後方法再返回。

  若是try塊中存在System.exit(0);語句,那麼就不會執行finally塊中的代碼,由於System.exit(0)會終止當前運行的Java虛擬機,程序會在虛擬機終止前結束執行。

相關文章
相關標籤/搜索