1、異常概述
在程序中,錯誤可能產(chǎn)生于程序員沒有料到的各種情況,或者是超出了程序員可控范圍的環(huán)境因素,如試圖打開一個不存在的文件等,在Java中,,這種在程序運行時可能出現(xiàn)的一些錯誤稱為異常。
2、異常的分類
Java類庫中每個包都定義了異常類,所有這些類都是Throwable類的子類,Throwable 類派生出兩個子類,分別是Error類和Exception類,其中,Error類及其子類用來描述Java運行系統(tǒng)中的內(nèi)部錯誤以及資源耗盡的錯誤,這類錯誤比較嚴重,而Exception類稱為非致命性類,可以通過捕捉處理使程序繼續(xù)執(zhí)行,Exception類又可以根據(jù)錯誤發(fā)生的原因分為運行時異常和非運行時異常。
2.1 系統(tǒng)錯誤-Error
Error類及其子類通常用來描述Java運行系統(tǒng)中的內(nèi)部錯誤,該類定義了常規(guī)環(huán)境下不希望由程序捕獲的異常,比如OutOfMemoryError,ThreadDeath等,這些錯誤發(fā)生時,Java一般會選擇線程終止。
2.2 異常-Exception
異常是程序本身可以處理的異常,這種異常主要分為運行時異常和非運行時異常,程序應(yīng)當(dāng)盡量去處理這些異常。
運行時異常
運行時異常是程序運行中產(chǎn)生的異常,這些異常一般由程序邏輯錯誤引起,程序應(yīng)當(dāng)從邏輯角度盡量避免這類異常的發(fā)生。
常見的運行時異常如下表所示:
異常類 | 說明 |
---|---|
ClassCastException | 類型轉(zhuǎn)換異常 |
NullPointerExcetion | 空指針異常 |
ArrayIndexOutOfBoundsException | 數(shù)組下標越界異常 |
ArithmeticException | 算術(shù)異常 |
ArrayStoreException | 數(shù)組中包含不兼容的值拋出的異常 |
NumberFormatException | 字符串轉(zhuǎn)換為數(shù)字時拋出的異常 |
IllegalArgumentException | 非法參數(shù)異常 |
FileSystemNotFoundException | 文件系統(tǒng)未找到異常 |
SecurityException | 安全性異常 |
StringIndexOutOfBoundsException | 字符串索引超出范圍拋出的異常 |
NegativeArraySizeException | 數(shù)組長度為負異常 |
非運行時異常
非運行時異常是RuntimeException類及其子類異常以外的異常,這類異常是必須進行處理的異常,如果不處理,程序就不能編譯通過,如IOException,SQLException以及用戶自定義的異常等,常見的非運行時異常有:
異常類 | 說明 |
---|---|
ClassNotFoundException | 未找到相應(yīng)類異常 |
SQLExcetion | 操作數(shù)據(jù)庫異常類 |
IOException | 輸入/輸出異常流 |
TimeoutException | 操作超時異常 |
FileNotFoundException | 文件未找到異常 |
3、捕捉處理異常
我們這里捕捉處理異常時,主要有兩種處理方式,一種是使用throws關(guān)鍵字,一種是使用try...catch代碼塊。
3.1 try..catch...finally代碼塊
package exceptionHandler;
public class FootballTeam {
public static void main(String[] args){
try{
String str = "lili";
System.out.println(str + "年齡是:" );
int age = Integer.parseInt("20L");
System.out.println(age);
}
catch(NumberFormatException e){
e.printStackTrace();
}
catch(Exception e){
e.printStackTrace();
}
finally{
System.out.println("program over");
}
}
}
輸出為
objc[85867]: Class JavaLaunchHelper is implemented in both /Library/Java/JavaVirtualMachines/jdk/Contents/Home/bin/java (0x10e0554c0) and /Library/Java/JavaVirtualMachines/jdk/Contents/Home/jre/lib/libinstrument.dylib (0x10e1314e0). One of the two will be used. Which one is undefined.
java.lang.NumberFormatException: For input string: "20L"
at java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)
at java.lang.Integer.parseInt(Integer.java:580)
at java.lang.Integer.parseInt(Integer.java:615)
at exceptionHandler.FootballTeam.main(FootballTeam.java:8)
lili年齡是:
program over
當(dāng)try中的語句跑出異常時,程序是不會繼續(xù)執(zhí)行try后面的語句的,相應(yīng)的,它會跳轉(zhuǎn)到執(zhí)行catch中的語句,而finally塊中的語句時無論有沒有異常都會正常執(zhí)行的。
另一方面,我們可以看到上面的程序中我們定義了兩個catch塊,也就是說,一個try可以對應(yīng)多個catch塊,每個catch塊捕獲一種特定的異常,但這里需要注意的是,catch塊的異常類順序必須先子類后父類,否則會發(fā)生錯誤。
3.2 在方法中拋出異常
如果某個方法可能會發(fā)生異常,但不想在當(dāng)前方法中處理這個異常,則可以使用throws、throw關(guān)鍵字在方法中拋出異常。
throws關(guān)鍵字
throws關(guān)鍵字通常被應(yīng)用在聲明方法時,用來指定方法可能拋出的異常,多個異常可以使用逗號間隔。
使用throws為方法拋出異常時,如果子類繼承父類,子類重寫方法拋出的異常也要和原父類方法拋出的異常相同或者是其異常的子類,除非throws異常是RuntimeException。
package exceptionHandler;
public class Shoot {
static void pop() throws NegativeArraySizeException {
int[] arr = new int[-3];
}
public static void main(String[] args){
try{
pop();
}
catch(NegativeArraySizeException e){
System.out.println("pop方法拋出的異常");
}
}
}
輸出為:
objc[85890]: Class JavaLaunchHelper is implemented in both /Library/Java/JavaVirtualMachines/jdk/Contents/Home/bin/java (0x1028394c0) and /Library/Java/JavaVirtualMachines/jdk/Contents/Home/jre/lib/libinstrument.dylib (0x1029154e0). One of the two will be used. Which one is undefined.
pop方法拋出的異常
throw關(guān)鍵字
throw關(guān)鍵字通常用于在方法體重制造一個異常,程序在執(zhí)行到throw語句時立即終止,它后面的語句都不執(zhí)行。
package exceptionHandler;
public class ThrowTest {
public static void main(String[] args){
int num1 = 25;
int num2 = 0;
int result;
if(num2 == 0){
throw new ArithmeticException("這都不會,小學(xué)生都知道了,除數(shù)不能是0");
}
result = num1 / num2;
System.out.println("兩個數(shù)的商為:" + result);
}
}
3.3 自定義異常
在程序中使用自定義異常類,大體可以分為以下幾個步驟:
1)創(chuàng)建自定義異常類
2)在方法中通過throw關(guān)鍵字拋出異常對象
3)如果在當(dāng)千拋出異常的方法中處理異常,可以使用try..catch代碼塊捕獲并處理,否則,在方法的聲明處通過throws關(guān)鍵字指名要拋給方法調(diào)用者的異常,繼續(xù)進行下一步操作。
4)在出現(xiàn)異常方法的調(diào)用這種捕獲并處理異常
package exceptionHandler;
class MyException extends Exception {
public MyException(String ErrorMessage){
super(ErrorMessage);
}
}
public class Tran {
static void avg(int age) throws MyException{
if(age < 0){
throw new MyException("年齡不可以使用負數(shù)");
}
else{
System.out.println("王師傅今年 " + age + " 歲了");
}
}
public static void main(String[] args){
try{
avg(-50);
}
catch(MyException e){
e.printStackTrace();
}
}
}