任何Java代碼都可以拋出異常,如:自己編寫的代碼、來自Java開發環境包中代碼,或者Java運行時系統。無論是誰,都可以通過Java的throw語句拋出異常。從方法中拋出的任何異常都必須使用throws子句。
1. throws拋出異常
如果一個方法可能會出現異常,但沒有能力處理這種異常,可以在方法聲明處用throws子句來聲明拋出異常。例如汽車在運行時可能會出現故障,汽車本身沒辦法處理這個故障,那就讓開車的人來處理。
throws語句用在方法定義時聲明該方法要拋出的異常類型,如果拋出的是Exception異常類型,則該方法被聲明為拋出所有的異常。多個異常可使用逗號分割。 throws語句的語法格式為:
methodname throws Exception1,Exception2,..,ExceptionN
{
}
方法名後的throws Exception1,Exception2,…,ExceptionN 為聲明要拋出的異常列表。當方法拋出異常列表的異常時,方法將不對這些類型及其子類類型的異常作處理,而拋向調用該方法的方法,由他去處理。例如:
import java.lang.Exception; public class TestException { static void pop() throws NegativeArraySizeException { // 定義方法並拋出NegativeArraySizeException異常int[] arr = new int[-3]; // 創建數組} public static void main( String[] args) { // 主方法try { // try語句處理異常信息pop(); // 調用pop()方法} catch (NegativeArraySizeException e) { System.out.println("pop()方法拋出的異常");// 輸出異常信息} } }
使用throws關鍵字將異常拋給調用者後,如果調用者不想處理該異常,可以繼續向上拋出,但最終要有能夠處理該異常的調用者。
pop方法沒有處理異常NegativeArraySizeException,而是由main函數來處理。
Throws拋出異常的規則:
1) 如果是不可查異常(unchecked exception),即Error、RuntimeException或它們的子類,那麼可以不使用throws關鍵字來聲明要拋出的異常,編譯仍能順利通過,但在運行時會被系統拋出。
2)必須聲明方法可拋出的任何可查異常(checked exception)。即如果一個方法可能出現受可查異常,要么用try-catch語句捕獲,要么用throws子句聲明將它拋出,否則會導致編譯錯誤
3)僅當拋出了異常,該方法的調用者才必須處理或者重新拋出該異常。當方法的調用者無力處理該異常的時候,應該繼續拋出,而不是囫圇吞棗。
4)調用方法必須遵循任何可查異常的處理和聲明規則。若覆蓋一個方法,則不能聲明與覆蓋方法不同的異常。聲明的任何異常必須是被覆蓋方法所聲明異常的同類或子類。
例如:
判斷一個方法可能會出現異常的依據如下:
1)方法中有throw語句。例如,以上method7()方法的catch代碼塊有throw語句。
2)調用了其他方法,其他方法用throws子句聲明拋出某種異常。例如,method3()方法調用了method1()方法,method1()方法聲明拋出IOException,因此,在method3()方法中可能會出現IOException。
2. 使用throw拋出異常
throw總是出現在函數體中,用來拋出一個Throwable類型的異常。程序會在throw語句後立即終止,它後面的語句執行不到,然後在包含它的所有try塊中(可能在上層調用函數中)從裡向外尋找含有與其匹配的catch子句的try塊。
我們知道,異常是異常類的實例對象,我們可以創建異常類的實例對象通過throw語句拋出。該語句的語法格式為:
throw new exceptionname;
例如拋出一個IOException類的異常對象:
throw new IOException;
要注意的是,throw 拋出的只能夠是可拋出類Throwable 或者其子類的實例對象。下面的操作是錯誤的:
throw new String(“exception”);
這是因為String 不是Throwable 類的子類。
如果拋出了檢查異常,則還應該在方法頭部聲明方法可能拋出的異常類型。該方法的調用者也必須檢查處理拋出的異常。
如果所有方法都層層上拋獲取的異常,最終JVM會進行處理,處理也很簡單,就是打印異常消息和堆棧信息。如果拋出的是Error或RuntimeException,則該方法的調用者可選擇處理該異常。
package Test; import java.lang.Exception; public class TestException { static int quotient(int x, int y) throws MyException { // 定義方法拋出異常if (y < 0) { // 判斷參數是否小於0 throw new MyException("除數不能是負數"); // 異常信息} return x/y; // 返回值} public static void main(String args[]) { // 主方法int a =3; int b =0; try { // try語句包含可能發生異常的語句int result = quotient(a, b); // 調用方法quotient() } catch (MyException e) { // 處理自定義異常System.out.println(e. getMessage()); // 輸出異常信息} catch (ArithmeticException e) { // 處理ArithmeticException異常System.out.println("除數不能為0"); // 輸出提示信息} catch (Exception e) { //處理其他異常System.out.println("程序發生了其他的異常"); // 輸出提示信息} } } class MyException extends Exception { // 創建自定義異常類String message; // 定義String類型變量public MyException (String ErrorMessagr) { // 父類方法message = ErrorMessagr; } public String getMessage() { // 覆蓋getMessage()方法return message; } }