詳細介紹Java關鍵字throw?throws?Throwable的用法與區(qū)別
throw,意為“投擲、拋、扔”。Throw、Throws和Throwable三者都用于異常處理。
1. Throwable
Throwable在Java中是異常處理這個分支的頂級父類,其它所有異常處理的實現(xiàn)都依賴于Throwable
打開Java官方文檔(Java8版本),找到Throwable,它的直接子類為Error和Exception。
Error和Exception兩者的特點在于Error異常程序無法處理,只能交由人工介入修改代碼,比如棧溢出、堆溢出等等;而Exception異??梢蕴崆鞍l(fā)覺,并作出有效處理。
1.1 擴展-Error
在Error中,常見的有棧溢出和堆溢出等等。
舉個例子,StackOverflowError
public class ErrorTest { public static void main(String[] args) { main(args); } }
無限遞歸,執(zhí)行這個程序就會報棧溢出異常。
再比如堆異常,OutOfMemoryError
public class ErrorTest { public static void main(String[] args) { Integer[] testArray = new Integer[1024*1024*1024]; } }
1.2 擴展-Exception
在Exception中就有非常多我們所熟知的異常情況了,比如NullPointerException(空指針異常)、ArrayIndexOutOfBoundsException(數(shù)組下標越界)、NumberFormatException(數(shù)字格式化異常)等等。
public class ExceptionTest { public static void main(String[] args) { int[] testArray = null; System.out.println(testArray[0]); //空指針異常 } }
public class ExceptionTest { public static void main(String[] args) { int[] testArray = new int[3]; System.out.println(testArray[3]); //數(shù)組下標越界 } }
public class ExceptionTest { public static void main(String[] args) { String num = "abc"; System.out.println(Integer.parseInt(num)); //數(shù)字格式化異常 } }
2. throws
throws應用在方法聲明處,指明此方法在執(zhí)行時可能會出現(xiàn)的異常類型。一旦該方法執(zhí)行時出現(xiàn)異常,就會在異常代碼處生成一個異常類的對象,此對象滿足Throws后的異常類型時,就會被拋出。這里有兩個過程,代碼有異常時
1. 生成一個異常對象;
2. throws捕獲到這個異常,將異常對象拋出
throws和try-catch-finally一起稱為異常處理的兩種方式。
try-catch-finally是在出現(xiàn)異常時主動處理掉異常,使得程序可以繼續(xù)執(zhí)行下去;而throws捕獲到異常之后向上拋出異常對象,不去真正地處理這個異常。
所謂向上拋出異常對象,是將異常對象交給調(diào)用者去處理,比如方法A調(diào)用方法B,B通過throws拋出異常,而A可以選擇使用try-catch-finally處理掉異常,也可以通過throws繼續(xù)向上拋出異常對象,直到異常被真正處理掉。如果一直沒有方法去處理異常,異常對象最終會被拋給JVM,從而導致程序停止運行。
@Test public void throwsTest(){ //調(diào)用者解決拋出的異常 try{ formatChange("abc"); } catch (NumberFormatException e){ System.out.println("轉(zhuǎn)換格式錯誤!"); } catch (Exception e){ System.out.println("出現(xiàn)錯誤"); } } private int formatChange(String str) throws NumberFormatException{ //出現(xiàn)異常向上拋出 return Integer.parseInt(str); }
2.1 擴展
——如何選擇try-catch-finally還是throws?
當一個方法中存在異常需要處理,在大多數(shù)情況下,既可以選擇try-catch-finally直接處理掉這個異常,也可以選擇throws向上拋出異常,交給調(diào)用者去處理(異常拋到最后,總要有一方真正地去處理這個異常,怎么處理?還是用try-catch-finally唄),在選擇上比較自由,但是,出現(xiàn)以下兩種情況時,需要遵循一定的規(guī)則(如有補充,敬請指出)。
- 如果父類中被重寫的方法沒有使用throws拋出異常,則子類重寫的方法也不能使用throws拋出異常,也就意味著這種情況必須使用try-catch-finally去處理。
- 在方法A中,先后調(diào)用了另外的幾種方法,這幾種方法是遞進關系執(zhí)行的且其中很多方法都存在異常需要處理,這種情況建議被調(diào)用的幾個方法使用throws向上拋出異常,在方法A中,使用try-catch-finally統(tǒng)一處理掉這些異常。
針對第一條,這是一個規(guī)定,子類中重寫的方法使用throws拋出的異常必須不大于父類中被重寫的方法拋出異常的范圍。舉個例子,父類中的方法B拋出NullPointerException異常,則子類中重寫B(tài)方法就不能拋出如Exception這種比NullPointerException范圍更大的異常;如果父類中被重寫的方法沒有拋出任何異常,則子類更不能拋出異常。
為什么?展示一段代碼。
//假設父類中的方法B拋出NullPointerException異常,子類中的方法B可以拋出Exception private void test(ParentClassTest parent){ try{ parent.B(); } catch(NullPointerException e){ System.out.println("出現(xiàn)了空指針異常"); } }
在本示例中,假設父類中的方法B拋出NullPointerException異常,子類中重寫的方法B可以拋出Exception。那么傳進給test方法的參數(shù)如果是父類的實例化對象,那么調(diào)用test方法沒有任何問題。如果傳進的參數(shù)是子類的實例化對象,再去調(diào)用子類重寫的方法B,那么就有可能拋出Exception異常,try-catch結構就壓不住這個異常了,這顯然是一個不合理的操作。
針對第二條,假設方法A中調(diào)用了方法C、D、E,這三個方法都有可能產(chǎn)生異常,且存在遞進關系,也就是D、E執(zhí)行需要C執(zhí)行完成、E執(zhí)行依賴C、D執(zhí)行完成。那么就推薦在C、D、E中向上拋出異常,在方法A中集中處理。為什么?如果C、D、E都是向上拋出異常,而A使用try-catch-finally去處理這個異常,如果某個方法真的出現(xiàn)異常,則不再繼續(xù)執(zhí)行。而如果C、D、E都使用try-catch-finally直接解決掉異常,那么即使產(chǎn)生了異常,方法A也不會接收到異常的產(chǎn)生,那么還會接著往下執(zhí)行,但是C出現(xiàn)了異常,再執(zhí)行D、E沒有任何意義。
3. throw
如果在程序編寫時有手動拋出異常的需求,則可以使用throw
throw使用在方法體內(nèi)。與try-catch-finally和throws都不同,異常處理的兩個階段:1.遇到異常,生成異常對象;2.捕獲到異常,進行拋出或處理。try-catch-finally和throws都處在第二個階段,都是捕獲到異常后的相關處理,一般使用系統(tǒng)根據(jù)異常類型自動生成的異常對象進行處理。而throw應用在第一階段,手動地產(chǎn)生一個異常對象。
舉一個例子,判斷一個數(shù)值是否為非負數(shù),如果為負數(shù),則拋出異常。
class ThrowTest{ private int Number; public void judge(int num){ if(num>=0){ this.Number = num; } else{ throw new RuntimeException("傳入?yún)?shù)為負數(shù)"); } } }
@Test public void test2(){ ThrowTest throwTest = new ThrowTest(); throwTest.judge(-100); }
成功拋出異常。
使用try-catch捕獲一下異常。
@Test public void test2(){ ThrowTest throwTest = new ThrowTest(); try{ throwTest.judge(-100); } catch (RuntimeException e){ System.out.println(e.getMessage()); } }
如果把throw拋出的異常改為Exception,則直接報錯,也就是不能編譯。Exception包含兩種異常:編譯時異常和運行時異常,前者在編譯前就要檢查是否有可能產(chǎn)生編譯時異常;后者是在編譯后運行時才會判斷的異常。而throw new Exception包含了編譯時異常,需要顯式處理掉這個異常,怎么處理?try-catch-finally或者throws
class ThrowTest{ private int Number; public void judge(int num) throws Exception{ if(num>=0){ this.Number = num; } else{ throw new Exception("傳入?yún)?shù)為負數(shù)"); } } }
調(diào)用方也要隨著進行更改。
@Test public void test2(){ ThrowTest throwTest = new ThrowTest(); try{ throwTest.judge(-100); } catch (RuntimeException e){ System.out.println(e.getMessage()); } catch (Exception e){ System.out.println(e.getMessage()); } }
3.1 擴展
——自定義異常類
throw還可以拋出自定義異常類。
自定義異常類的聲明需要繼承于現(xiàn)有的異常體系。
class MyException extends RuntimeException{ static final long serialVersionUID = -703489719076939L; //可以認為是一種標識 public MyException(){} public MyException(String message){ super(message); } }
此時我們可以拋出自定義的異常
class ThrowTest{ private int Number; public void judge(int num) throws MyException{ if(num>=0){ this.Number = num; } else{ throw new MyException("不能輸入負數(shù)"); } } }
調(diào)用者修改
@Test public void test2(){ ThrowTest throwTest = new ThrowTest(); try{ throwTest.judge(-100); } catch (MyException e){ System.out.println(e.getMessage()); } }
4. 總結
三者共同點在于都屬于是異常處理的范疇內(nèi)。
不同點:
- Throwable是異常處理這個分支的頂層父類,其它異常類的實現(xiàn)都需要繼承于Throwable
- throw應用在方法體內(nèi),是生成異常對象的一種方式
- throws應用在方法聲明處,聲明出可能要拋出的各種異常類,是處理異常的方式
到此這篇關于詳細介紹Java關鍵字throw throws Throwable的用法與區(qū)別的文章就介紹到這了,更多相關Java關鍵字throw throws Throwable內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
MyBatis Plus 將查詢結果封裝到指定實體的方法步驟
這篇文章主要介紹了MyBatis Plus 將查詢結果封裝到指定實體的方法步驟,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-09-09tk.mybatis實現(xiàn)uuid主鍵生成的示例代碼
本文主要介紹了tk.mybatis實現(xiàn)uuid主鍵生成的示例代碼,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-12-12spring boot環(huán)境抽象的實現(xiàn)方法
在實際開發(fā)中,開發(fā)人員在編寫springboot的時候通常要在本地環(huán)境測試然后再部署到Production環(huán)境,這兩種環(huán)境一般來講是不同的,最主要的區(qū)別就是數(shù)據(jù)源的不同。本文主要介紹了這兩種,感興趣的可以了解一下2019-04-04Spring Boot下如何自定義Repository中的DAO方法
這篇文章主要介紹了Spring Boot下如何自定義Repository中的DAO方法,需要的朋友可以參考下2017-06-06