概述
在運行的時候發(fā)生不正常的情況。在Java中采用類的形式對異常問題進行描述和封裝對象空猜。
對比Exception和Error
Exception 和 Error 都是繼承了 Throwable 類,在 Java 中只有 Throwable 類型的實例才可以被拋出(throw)或者捕獲(catch),它是異常處理機制的基本組成類型。
Exception 是程序正常運行中,可以預料的意外情況朋魔,可能并且應該被捕獲,進行相應處理卿操。
Error 是指在正常情況下警检,不大可能出現(xiàn)的情況,絕大部分的 Error 都會導致程序(比如 JVM 自身)處于非正常的害淤、不可恢復狀態(tài)扇雕。既然是非正常情況,所以不便于也不需要捕獲窥摄,常 見的比如 OutOfMemoryError 之類镶奉,都是 Error 的子類。
Throwable中的常用方法:
- public void printStackTrace():打印異常的詳細信息崭放。
包含了異常的類型,異常的原因,還包括異常出現(xiàn)的位置,在開發(fā)和調(diào)試階段,都得使用printStackTrace哨苛。
- public String getMessage():獲取發(fā)生異常的原因。
提示給用戶的時候,就提示錯誤原因莹菱。
- public String toString():獲取異常的類型和異常描述信息(不用)移国。
異常分類
可檢查(checked)異常,
可檢查異常在源代碼里必須顯式地進行捕獲處理道伟,這是編譯期檢查的一部分。前面介紹的不可查的 Error ,是 Throwable 不是 Exception 蜜徽。
不檢查 (unchecked)異常
不檢查異常就是所謂的運行時異常祝懂,類似 NullPointerException ArrayIndexOutOfBoundsException之類,通常是可以編碼避免的邏輯錯誤拘鞋,具體根據(jù)需要來判斷是否需要捕獲砚蓬,并不會在編譯期強制要求。
try-catch代碼段會產(chǎn)生額外的性能開銷盆色,或者換個角度說灰蛙,它往往會影響JVM對代碼進行優(yōu)化,所以建議僅捕獲有必要的代碼段隔躲,盡量不要一個大的try包住整段的代碼摩梧;與此同時,利用異承担控制代碼流程仅父,也不是一個好主意,遠比我們通常意義上的條件語句( if/else 浑吟、 switch )要低效笙纤。
Java每實例化一個Exception,都會對當時的棧進行快照组力,這是一個相對比較重的操作省容。如果發(fā)生的非常頻繁,這個開銷可就不能被忽略了燎字。
所以蓉冈,對于部分追求極致性能的底層類庫,有種方式是嘗試創(chuàng)建不進行椥ィ快照的 Exception 寞酿。這本身也存在爭議,因為這樣做的假設在于脱柱,我創(chuàng)建異常時知道未來是否需要堆棧伐弹。問 題是,實際上可能嗎榨为?小范圍或許可能惨好,但是在大規(guī)模項目中,這么做可能不是個理智的選擇随闺。如果需要堆棧日川,但又沒有收集這些信息,在復雜情況下矩乐,尤其是類似微服務這種分布 式系統(tǒng)龄句,這會大大增加診斷的難度回论。
常見異常
1、NullPointerException
空指針異常分歇,操作一個 null 對象的方法或屬性時會拋出這個異常傀蓉。
public static void main(String[] args) {
int[] arr = null;
System.out.println(arr[1]);
}
2、OutOfMemoryError
內(nèi)存異常異常职抡,這不是程序能控制的葬燎,是指要分配的對象的內(nèi)存超出了當前最大的堆內(nèi)存,需要調(diào)整堆內(nèi)存大懈克Α(-Xmx)以及優(yōu)化程序谱净。
3、IOException
IO擅威,即:input, output壕探,我們在讀寫磁盤文件、網(wǎng)絡內(nèi)容的時候經(jīng)常會生的一種異常裕寨,這種異常是受檢查異常浩蓉,需要進行手工捕獲。
如文件讀寫會拋出 IOException:
public int read() throws IOExceptionpublic void write(int b) throws IOException
4宾袜、FileNotFoundException
文件找不到異常捻艳,如果文件不存在就會拋出這種異常。如定義輸入輸出文件流庆猫,文件不存在會報錯,FileNotFoundException 其實是 IOException 的子類认轨,同樣是受檢查異常,需要進行手工捕獲月培。
5嘁字、ClassNotFoundException
類找不到異常,Java開發(fā)中經(jīng)常遇到杉畜,是不是很絕望纪蜒?這是在加載類的時候拋出來的,即在類路徑下不能加載指定的類此叠〈啃看一個示例:
它是受檢查異常,需要進行手工捕獲灭袁。
** 6猬错、ClassCastException**
類轉換異常,將一個不是該類的實例轉換成這個類就會拋出這個異常茸歧。
如將一個數(shù)字強制轉換成字符串就會報這個異常:
public static void main(String[] args) {
Object a = 1;
String s = (String) a;
System.out.println(s);
}
這是運行時異常倦炒,不需要手工捕獲。
7软瞎、NoSuchMethodException
沒有這個方法異常逢唤,一般發(fā)生在反射調(diào)用方法的時候
8拉讯、IndexOutOfBoundsException 越界異常
索引越界異常,當操作一個字符串或者數(shù)組的時候經(jīng)常遇到的異常智玻。
public static void main(String[] args) {
int[] arr = {1,2,3,4};
System.out.println(arr[4]);
}
** 9遂唧、ArithmeticException**
算術異常芙代,發(fā)生在數(shù)字的算術運算時的異常吊奢,如一個數(shù)字除以 0 就會報這個錯。
double n = 3 / 0;
10纹烹、SQLException
SQL異常页滚,發(fā)生在操作數(shù)據(jù)庫時的異常。
如下面的獲取連接:
又或者是獲取下一條記錄的時候:
boolean next() throws SQLException;
它是受檢查異常铺呵,需要進行手工捕獲裹驰。
異常處理
捕獲異常try...catch...finally
捕獲異常語法格式如下,try和catch都不能單獨使用,必須連用。finally是可選性
try{
可能出現(xiàn)異常代碼
}catch(Exception e){
處理異常的代碼
}finally{
一定會執(zhí)行的代碼
}
eg:
public static void main(String[] args) {
try{
int a = 2/0;
}catch (ArithmeticException e){
System.out.println("除數(shù)不能為0");
}
}
多個錯誤的處理
public static void main(String[] args) {
try {
int[] arr = {1,2,3,4};
System.out.println(arr[4]);//ArrayIndexOutOfBoundsException數(shù)組越界異常
int a = 2 / 0;//ArithmeticException:算術異常
System.out.println(a);
int[] arr1 = null;
System.out.println(arr1[0]);//NullPointerException空指針
} catch (ArrayIndexOutOfBoundsException e) {
System.out.println("數(shù)組越界異常...");
} catch (ArithmeticException e){
System.out.println("算術異常...");
} catch(NullPointerException e){
System.out.println("空指針異常...");
}
}
多個異常通過|連接
try {
} catch (ArrayIndexOutOfBoundsException |ArithmeticException | NullPointerException e) {
System.out.println(e.getClass());
}
finaly和return
public static void main(String[] args) {
System.out.println(divFun(2,0));
}
public static int divFun(int a,int b){
int result = 0;
try {
result = a/b;
} catch (Exception e) {
System.out.println(e.getClass());
result = -1;
return result;
}finally {
System.out.println("finally action");
}
return result;
}
聲明異常throws
使用throws聲明的方法表示此方法不處理異常片挂,拋給方法的調(diào)用者處理,用在方法聲明后面幻林,跟的是異常類名,可以跟多個異常類名,用逗號隔開音念。關鍵字throws運用于方法聲明之上,用于表示當前方法不處理異常,而是提醒該方法的調(diào)用者來處理異常(拋出異常).
修飾符 返回值類型 方法名(參數(shù)) throws 異常類名1,異常類名2…{ }
eg
public static void main(String[] args) {
try {
System.out.println(divFun(2,0));
}catch (Exception e){
System.out.println(e);
}
}
public static int divFun(int a,int b) throws ArithmeticException{
int result = a/b;
return result;
}
它表示拋出異常沪饺,由該方法的調(diào)用者來處理
拋出異常throw
用在方法體內(nèi),跟的是異常對象名,拋出的時候直接拋出異常類的實例對象闷愤。將這個異常對象傳遞到調(diào)用者處整葡,并結束當前方法的執(zhí)行。
throw new 異常類名(參數(shù));
eg:
public static int divFun(int a, int b) {
if (b == 0) {
throw new ArithmeticException();
}
int result = a / b;
return result;
}
自定義異常
自定義異常直接繼承Exception就可以完成自定義異常類
public static int divFun(int a, int b) throws ZeroException {
if (b == 0) {
throw new ZeroException("除數(shù)不能為0");
}
int result = a / b;
return result;
}
class ZeroException extends Exception{
public ZeroException(String message){
super(message);
}
}