Java 異常詳解
一.異常與錯(cuò)誤的區(qū)別
再講異常之前我們就應(yīng)該要知道異常和錯(cuò)誤的區(qū)別
Error類和Exception類的父類都是throwable類,他們的區(qū)別是:
Error類一般是指與虛擬機(jī)相關(guān)的問題,如系統(tǒng)崩潰,虛擬機(jī)錯(cuò)誤,內(nèi)存空間不足,方法調(diào)用棧溢等。對(duì)于這類錯(cuò)誤的導(dǎo)致的應(yīng)用程序中斷,
僅靠程序本身無法恢復(fù)和和預(yù)防,遇到這樣的錯(cuò)誤,建議讓程序終止。
Exception類表示程序可以處理的異常,可以捕獲且可能恢復(fù)。遇到這類異常,應(yīng)該盡可能處理異常,使程序恢復(fù)運(yùn)行,而不應(yīng)該隨意終止異常。
二.異常的體現(xiàn)分類:
1.checked 異常檢查期異常 java.lang.Excetion 在編譯期需要人工處理否則編譯失?。篍xception的子類除了運(yùn)行期異常都是檢查期異常
2.非Checked異常運(yùn)行時(shí)異常 java.lang.RuntimeException 不需要處理的直接能編譯通過:所有的RuntimeException以及其子類都是運(yùn)行異常
舉例:運(yùn)行期異常
結(jié)果:運(yùn)行期異常,當(dāng)你敲好代碼時(shí)不會(huì)報(bào)錯(cuò),而當(dāng)你運(yùn)行時(shí)才會(huì)報(bào)除數(shù)不能為0的錯(cuò)誤
舉例:檢查期異常:
結(jié)果:檢查期異常,當(dāng)你編譯的時(shí)候就會(huì)報(bào)錯(cuò),一定要拋出異常編譯才能通過
三.異常的處理機(jī)制
Java語言主要依賴于 try catch finally 和throws throw 五個(gè)關(guān)鍵字來描述異常
1) 在發(fā)生異常的地方直接處理
使用try catch finally 直接處理異常
a) try-catch-finally結(jié)構(gòu)中try塊是必須有的,catch和finally塊為可選,但兩者至少必須出現(xiàn)其中之一。
b) catch 可以同時(shí)出現(xiàn)多個(gè),但一個(gè)異常最多捕獲一個(gè)catch,而且catch的順序是從上往下
c) finally 無論是否捕獲異常都會(huì)執(zhí)行的一行代碼
演示1:try異常
public class TestException { public static void main(String[] args) { int c = 0; try { int a = 3; int b = 0; // 這塊代碼出現(xiàn)了異常 c = a / b; // 那么異常之后的代碼都不會(huì)被執(zhí)行 System.out.println("Hello World"); } catch (ArithmeticException e) { System.out.println("除數(shù)不能為零"); } finally { //不管有沒有發(fā)生異常,finally語句塊都會(huì)被執(zhí)行 System.out.println("Welcome"); } System.out.println(c); // 當(dāng)b為0時(shí),有異常,輸出為c的初始值0 } } //輸出結(jié)果:除數(shù)不能為零 Welcome 0 try異常
演示2:帶有return的異常
import java.io.FileInputStream; import java.io.FileNotFoundException; public class DemoException { public static void main(String[] args) { int a=test3(); System.out.println(a); } @SuppressWarnings("finally") public static int test3(){ try { System.out.println(9 / 0); return 1; } catch (Exception e) { System.out.println("捕獲到了異常...."); return 2; }finally{ System.out.println("無論如何都會(huì)執(zhí)行的代碼..."); return 3; } } } //輸出結(jié)果 "呵呵""哈哈" 3 帶有return異常
得出結(jié)論:作用范圍 return 終止整個(gè)方法體,但在finally出現(xiàn)之前 return是老大 finally 作用范圍> return
2)將異常拋給調(diào)用者讓調(diào)用者處理
//throws在方法體頭部通過聲明 拋出異常... public void dealFile() throws FileNotFoundException{ FileInputStream fis =new FileInputStream("C:/name.txt"); } //那么那么上面調(diào)用這個(gè)方法可以選擇是繼續(xù)拋出,還是捕獲異常
案例一:通過throws拋出異常,調(diào)用者直接捕獲拋出的異常
public class TestException { public static void main(String[] args) { try { Test3(); //這里選擇直接捕獲異常,而不是在拋出異常 } catch (NumberFormatException e) { System.err.println("非數(shù)據(jù)類型不能轉(zhuǎn)換。"); } //System.err.println();這種輸出方式可以輸出錯(cuò)誤的消息,在控制臺(tái)呈現(xiàn)紅色。 } public static void Test3() throws NumberFormatException{ String s = "abc"; System.out.println(Double.parseDouble(s)); } } throws異常
運(yùn)行結(jié)果:
非數(shù)據(jù)類型不能轉(zhuǎn)換。
注意:使用Throws是的限制
兩小原則
使用throws 聲明拋出異常一個(gè)限制
子類繼承父類重寫父類的方法
子類拋出的異常必須比父類少
子類拋出的異常必須比父類小
兩小原則是針對(duì)檢查期異常的,運(yùn)行期異常不遵循這個(gè)規(guī)則(RuntimeException 以及子類)
案例二:通過throw拋出異常
public class TestException { public static void main(String[] args) { String s = "abc"; if(s.equals("abc")) { throw new NumberFormatException("不能相等"); } else { System.out.println(s); } } } throw異常
運(yùn)行結(jié)果如下:
面試題:Throw 和Throws有什么區(qū)別?
Throw語句是用在方法體內(nèi)表示拋出的異常由方法體內(nèi)的語句處理
Throws 語句是在方法聲明之后拋出異常表示在拋出的異常交給調(diào)用者處理
Throws 要么使用try –catch –finally 處理要么繼續(xù)拋出
四.自定義異常
所謂自定義異常,通常就是定義一個(gè)類,去繼承Exception類或者它的子類。因?yàn)楫惓1仨氈苯踊蛘唛g接地繼承自Exception類。
通常情況下,會(huì)直接繼承自Exception類,一般不會(huì)繼承某個(gè)運(yùn)行時(shí)的異常類。
自定義異??梢杂糜谔幚碛脩舻卿涘e(cuò)誤,用戶輸入錯(cuò)誤提示等。
自定義異常需要遵循以下步驟
- 繼承RuntimeExcetion 或者Exception
- 寫一個(gè)無參的構(gòu)造函數(shù)
- 寫一個(gè)String類型的構(gòu)造函數(shù)
舉例:自定義異常:
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("傳入的字符串參數(shù)不能為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("傳入的字符串參數(shù)不能為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("程序處理完畢"); } } }
最后說一句,try-catch-finally雖好用,但是如果是濫用,這樣只是會(huì)讓程序的可讀性變的很糟糕,當(dāng)程序報(bào)錯(cuò),就無法快速準(zhǔn)確的定位了。
以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來一定的幫助,同時(shí)也希望多多支持腳本之家!
相關(guān)文章
SpringBoot獲取maven打包時(shí)間的兩種方式
這篇文章主要介紹了SpringBoot獲取maven打包時(shí)間的兩種方式,文章通過代碼示例給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-05-05Java如何按16進(jìn)制發(fā)送和接收TCP指令
這篇文章主要介紹了Java如何按16進(jìn)制發(fā)送和接收TCP指令問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09Mybatis把返回結(jié)果封裝成map類型的實(shí)現(xiàn)
本文主要介紹了Mybatis把返回結(jié)果封裝成map類型的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03淺談Mybatis+mysql 存儲(chǔ)Date類型的坑
這篇文章主要介紹了淺談Mybatis+mysql 存儲(chǔ)Date類型的坑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-11-11java如何實(shí)現(xiàn)項(xiàng)目啟動(dòng)時(shí)執(zhí)行指定方法
這篇文章主要為大家詳細(xì)介紹了java項(xiàng)目如何啟動(dòng)時(shí)執(zhí)行指定方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07Java讀取resources中資源文件路徑以及jar中文件無法讀取的解決
這篇文章主要介紹了Java讀取resources中資源文件路徑以及jar中文件無法讀取的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05SpringBoot中間件ORM框架實(shí)現(xiàn)案例詳解(Mybatis)
這篇文章主要介紹了SpringBoot中間件ORM框架實(shí)現(xiàn)案例詳解(Mybatis),本篇文章提煉出mybatis最經(jīng)典、最精簡(jiǎn)、最核心的代碼設(shè)計(jì),來實(shí)現(xiàn)一個(gè)mini-mybatis,從而熟悉并掌握ORM框架的涉及實(shí)現(xiàn),需要的朋友可以參考下2023-07-07java開發(fā)Dubbo注解Adaptive實(shí)現(xiàn)原理
這篇文章主要為大家介紹了java開發(fā)Dubbo注解Adaptive實(shí)現(xiàn)原理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09