Java異常分為兩種類型昧廷,checked異常和unchecked異常,另一種叫法是異常和錯(cuò)誤甚侣。
簡(jiǎn)單地講,checked是可以在執(zhí)行過(guò)程中恢復(fù)的间学,而unchecked異常則是無(wú)法處理的錯(cuò)誤殷费。
checked異常:
1、表示無(wú)效低葫,不是程序中可以預(yù)測(cè)的详羡。比如無(wú)效的用戶輸入,文件不存在嘿悬,網(wǎng)絡(luò)或者數(shù)據(jù)庫(kù)鏈接錯(cuò)誤实柠。這些都是外在的原因,都不是程序內(nèi)部可以控制的鹊漠。
2主到、必須在代碼中顯式地處理茶行。比如try-catch塊處理躯概,或者給所在的方法加上throws說(shuō)明,將異常拋到調(diào)用棧的上一層畔师。
3娶靡、繼承自java.lang.Exception(java.lang.RuntimeException除外)。
unchecked異常:
1看锉、表示錯(cuò)誤姿锭,程序的邏輯錯(cuò)誤。是RuntimeException的子類伯铣,比如IllegalArgumentException, NullPointerException和IllegalStateException呻此。
2、不需要在代碼中顯式地捕獲unchecked異常做處理腔寡。
3焚鲜、繼承自java.lang.RuntimeException(而java.lang.RuntimeException繼承自java.lang.Exception)。
Java中的checked異常需要在代碼中顯式地通過(guò)try-catch捕獲或者再拋出,如果不需要對(duì)這個(gè)異常做處理忿磅,可以簡(jiǎn)地將異常再次拋出糯彬,這種異常存在一點(diǎn)不足,很多人習(xí)慣在代碼中直接寫(xiě)一個(gè)空的catch塊葱她,這樣不但使代碼變得有些冗余和“丑陋”撩扒,而且給調(diào)試帶來(lái)麻煩,增加了代碼維護(hù)的難度吨些。所以有人說(shuō)checked使代碼變的冗長(zhǎng)搓谆,空的catch塊也沒(méi)什么意義,所以checked異常應(yīng)該從Java標(biāo)準(zhǔn)中去除豪墅,像C#中就沒(méi)有checked異常的概念挽拔,C#中不強(qiáng)制顯式地捕獲異常。
Java異常之所以會(huì)分為這兩種但校,應(yīng)該是出于如下考慮:
checked異丑ψ纾可以幫助開(kāi)發(fā)人員意識(shí)到哪一行有可能會(huì)出現(xiàn)異常,因?yàn)镴ava的API已經(jīng)說(shuō)明了調(diào)用哪些方法可能會(huì)拋出異常状囱。如果不做處理編譯就不能通過(guò)术裸,從某種程度上說(shuō),這種做法可以避免程序的一些錯(cuò)誤亭枷。
兩個(gè)簡(jiǎn)單的例子:
import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
public class Main {
public static void main(String[] args) {
File f = new File("C:\test.txt");
FileReader r = new FileReader(f); //A
BufferedReader br = new BufferedReader(r);
br.readLine(); //B
br.close(); //C
}
}
這段代碼不能通過(guò)編譯袭艺,原因是A,B和C行會(huì)拋出IOException叨粘,必須將這段代碼放入try-catch塊中猾编,或者給main方法加上throws IOException才能編譯。
public class Main {
public static void main(String[] args) {
int a = 0;
int b = 100;
int c = b/a;
}
}
可以編譯升敲,但執(zhí)行會(huì)報(bào)錯(cuò)
Exception in thread “main” java.lang.ArithmeticException: / by zero at Main.main(Main.java:13)
ArithmeticException是一個(gè)unchecked異常答倡。
自定義異常
checked異常
自定義異常類InvalidUrlException
public class InvalidUrlException extends Exception {
public InvalidUrlException(String s){
super(s);
}
}
public class Main {
public static void getRemoteData(String url) throws InvalidUrlException{
if(isValidUrl(url)){
//獲取遠(yuǎn)程數(shù)據(jù)
}
else {
throw new InvalidUrlException("Invalid URL: " + url);
}
}
public static boolean isValidUrl(String url){
.... //驗(yàn)證URL是否有效
}
public static void main(String[] args) {
getRemoteData(args[0]);
}
}
如果在main方法里調(diào)用getRemoteData,有兩種做法驴党,一種是try-catch瘪撇,另一種則是直接給main加上throws InvalidUrlException。
unchecked異常
如果將InvalidUrlException改成extends RuntimeException港庄,那么main不需要再加throws或者try-catch倔既。
選擇checked還是unchecked異常?
一些Java書(shū)籍建議使用checked異常處理程序中所有可恢復(fù)的異常鹏氧,而用unchecked異常作為無(wú)法恢復(fù)的錯(cuò)誤渤涌。但是實(shí)際上那些繼承自RuntimeException的Java異常大多也可以在程序中恢復(fù)的,比如NullPointerException把还、IllegalArgumentExceptions实蓬、除0異常等等都可以通過(guò)捕獲處理使程序繼續(xù)運(yùn)行稿存。只有一些特殊情況會(huì)破壞程序的執(zhí)行,比如啟動(dòng)的時(shí)候讀取配置文件瞳秽,如果配置文件不存在或者有嚴(yán)重錯(cuò)誤瓣履,程序只好退出。
以下是是支持和反對(duì)checked異常的一些觀點(diǎn):
1练俐、編譯器強(qiáng)制捕獲或者拋出unchecked異常使開(kāi)發(fā)人員時(shí)刻記著要處理異常袖迎。
2、拋出checked異常的方法必須聲明throws腺晾,throws成為了方法或者接口的一部分燕锥,給后續(xù)版本增加或者修改方法的異常帶來(lái)不便。
3悯蝉、unchecked異常不需要顯式地處理反而使異常處理變的困難归形。
4、調(diào)用checked異常的方法鼻由,就必須處理這個(gè)方法的異常暇榴,這使得上層的調(diào)用者代碼混亂。
選擇checked還是unchecked都是由你自己決定的蕉世,很難說(shuō)哪種就一定正確蔼紧,反之就一定錯(cuò)誤。目前比較流行的語(yǔ)言中狠轻,Java似乎是唯一支持checked異常的語(yǔ)言奸例,其他語(yǔ)言都只有unchecked異常。