欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

帶你了解Java中的異常處理(下)

 更新時間:2020年08月18日 08:56:52   作者:弗蘭克的貓  
這篇文章主要介紹了Java中的異常處理的相關資料,幫助大家更好的理解和學習Java,感興趣的朋友可以了解下

  今天繼續(xù)講解java中的異常處理機制,主要介紹Exception家族的主要成員,自定義異常,以及異常處理的正確姿勢。

Exception家族

  一圖勝千言,先來看一張圖。

  Exception這是一個父類,它有兩個兒子,IOException和RuntimeException,每個兒子都很能生,所以它有著一堆的孫子,但其實,Exception家族還有一個大家伙,那就是Throwable,這是一個接口,看名字就知道意思,就是“可被拋出”嘛,它還有一個同父異母的哥哥,那就是Error,這家伙可厲害了,Error類一般是指與虛擬機相關的問題,如系統(tǒng)崩潰,虛擬機錯誤,內存空間不足,方法調用棧溢出等。catch語句里,不僅可以catch住Exception,還能catch住Error(什么?你真的打算catch Error??程獨秀同學,你先坐下。)一般情況下,是不能捕獲Error的,對于這類錯誤,Java編譯器不去檢查他們。對于這類錯誤的導致的應用程序中斷,僅靠程序本身無法恢復和預防,遇到這樣的錯誤,建議讓程序終止。除非你有把握能正確處理,否則程獨秀同學還是坐下吧(滑稽)。

Unchecked Exception和Checked Exception

  你也許會一臉懵逼,???,這是啥?異常也是分派別的,Unchecked Exception表示“未檢查異?!埃珻hecked Exception自然就是”已檢查異?!埃缮贓rror或者RuntimeException的異常稱為unchecked異常,所有其他的異常成為checked異常。那問題來了,為啥要區(qū)分這兩種異常?

  我們可以再看看上面那個圖,可以看出,RuntimeException和Error都是由程序內部引發(fā)的錯誤,比如上一篇里所說的空指針和算術異常。而Checked Exception則大都是由外部因素導致的,如文件無法找到異常,這是虛擬機無法掌控的情況,當出現異常,虛擬機也只能一臉懵逼,不知道該如何是好,所以當有可能發(fā)生時,就必須要使用try..catch去捕獲它,而對于Unchecked Exception 時,大部分是由于代碼引發(fā)的,所以只要代碼寫的足夠完善,是不會拋出這樣的異常的,所以也不強制要求捕獲。

  所以原因其實很簡單,編譯器將檢查你是否為所有的已檢查異常提供了異常處理機制,比如說我們使用Class.forName()來查找給定的字符串的class對象的時候,如果沒有為這個方法提供異常處理,編譯是無法通過的。已檢查異常的意義就在于讓你知道,這地方是有可能拋異常的,你要注意了,趕緊捕獲了。

自定義異常

  那么如何自定義一個異常呢?其實很簡單,只需要繼承Exception類就好了。看下面的栗子:

public class MyException extends Exception {
 public MyException() {
  super();
 }

 public MyException(String message) {
  super(message);
 }

 public MyException(String message, Throwable cause) {
  super(message, cause);
 }

 public MyException(Throwable cause) {
  super(cause);
 }

 protected MyException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {
  super(message, cause, enableSuppression, writableStackTrace);
 }
}

  MyException繼承了Exception類,重寫了構造函數,并沒有加自己的邏輯,只是調用了父類的方法。你看,自定義一個異常其實很簡單吧。看到這你也許又疑惑了,這尼瑪好像就是給Exception換了個名字,有啥用???

  別急,別急,你忘了嗎,Exception不僅是可以捕獲的,還是可以主動拋出的,所以當遇到某些特定的情況時,我們就可以主動拋出異常,然后在調用時去捕獲它,獲取異常信息,如果直接用Exception的話,那么捕獲的時候,會把所有的異常,該捕獲不該捕獲的都一起捕獲了,那么就沒法區(qū)分哪些是我們主動拋出來的異常了,這樣就無法對那些異常進行特殊處理了。

異常處理的正確姿勢  

  接下來要簡單介紹一個實際使用中常用的異常處理方法——異常鏈化處理。

  在一些大型的,模塊化的軟件開發(fā)中,一旦一個地方發(fā)生異常,則如骨牌效應一樣,將導致出現一連串的異常。假設B模塊需要調用A模塊的方法,如果A模塊發(fā)生異常,則B也將不能完成而發(fā)生異常,但是B在拋出異常時,會將A的異常信息掩蓋掉,這將使得異常的根源信息丟失。而使用異常的鏈化可以將多個模塊的異常串聯(lián)起來,使得異常信息不會丟失。

  異常鏈化就是用一個異常對象為參數構造新的異常對象。新的異對象將包含先前異常的信息。這項技術主要是異常類的一個帶Throwable參數的函數來實現的。這個當做參數的異常,我們叫他根源異常(cause)。如果你細心一點的話,會發(fā)現上面的栗子里也有一個叫做cause的東西,沒錯,說的其實就是它,在new一個新的異常時,將之前的異常信息傳入構造函數即可。下面再用一個簡單的栗子進行說明:

public class Test {
 public static void main(String[] args) {
  System.out.println("請輸入2個加數");
  int result;
  try {
   result = add();
   System.out.println("結果:"+result);
  } catch (Exception e){
   e.printStackTrace();
  }
 }

 /**
  * 執(zhí)行加法計算
  */
 private static int add() throws Exception {
  int result;
  try {
   List<Integer> nums =getInputNumbers();
   result = nums.get(0) + nums.get(1);
  }catch(InputMismatchException immExp){
   //鏈化:以一個異常對象為參數構造新的異常對象。
   throw new Exception("計算失敗",immExp);
  }
  return result;
 }

 /**
  * 獲取輸入的整數
  */
 private static List<Integer> getInputNumbers() {
  List<Integer> nums = new ArrayList<>();
  Scanner scan = new Scanner(System.in);
  try {
   int num1 = scan.nextInt();
   int num2 = scan.nextInt();
   nums.add(new Integer(num1));
   nums.add(new Integer(num2));
  }catch(InputMismatchException immExp){
   throw immExp;
  }finally {
   scan.close();
  }
  return nums;
 }
}

  輸出如下:

請輸入2個加數
d d
java.lang.Exception: 計算失敗
at com.frank.chapter17.Test.add(Test.java:35)
at com.frank.chapter17.Test.main(Test.java:18)
Caused by: java.util.InputMismatchException
at java.util.Scanner.throwFor(Scanner.java:864)
at java.util.Scanner.next(Scanner.java:1485)
at java.util.Scanner.nextInt(Scanner.java:2117)
at java.util.Scanner.nextInt(Scanner.java:2076)
at com.frank.chapter17.Test.getInputNumbers(Test.java:47)
at com.frank.chapter17.Test.add(Test.java:31)
... 1 more

  可以看到,當輸入的不是整數時,發(fā)生了異常,在getInputNumbers方法里沒有處理這個異常,而是將它繼續(xù)拋出,在add方法里捕獲了異常之后,以該異常為構造參數,重新拋出了一個異常,從打印輸出的信息可以看到,不僅僅有第二次拋出的異常信息,第一次的輸出信息不匹配異常的詳細信息也包含在了里面,銜接在Caused by之后,形成了一條異常鏈,這樣可以方便我們更快的排查問題所在。

  至此,異常就講解完畢了,希望能給大家?guī)硪恍﹩l(fā)和思考,如果覺得還算ok的話,記得動動小手點推薦,讓更多人可以看到,也歡迎關注我的博客,會持續(xù)更新的。如果有什么講的不好的地方。。。emmmmmm,你倒是來打我呀(逃)

以上就是帶你了解Java中的異常處理(下)的詳細內容,更多關于Java 異常處理的資料請關注腳本之家其它相關文章!

相關文章

  • JAVA-4NIO之Channel之間的數據傳輸方法

    JAVA-4NIO之Channel之間的數據傳輸方法

    下面小編就為大家?guī)硪黄狫AVA-4NIO之Channel之間的數據傳輸方法。小編覺得挺不錯的,現在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-06-06
  • Spring Boot Admin 的使用詳解

    Spring Boot Admin 的使用詳解

    這篇文章主要介紹了Spring Boot Admin 的使用詳解,Spring Boot Admin 用于監(jiān)控基于 Spring Boot 的應用,有興趣的可以了解一下
    2017-09-09
  • SpringBoot應用部署到外置Tomcat的實現

    SpringBoot應用部署到外置Tomcat的實現

    SpringBoot內置tomcat使用很方便,本文主要介紹了SpringBoot應用部署到外置Tomcat的實現,具有一定的參考價值,感興趣的可以了解一下
    2024-07-07
  • SpringBoot定時任務動態(tài)擴展ScheduledTaskRegistrar詳解

    SpringBoot定時任務動態(tài)擴展ScheduledTaskRegistrar詳解

    這篇文章主要為大家介紹了SpringBoot定時任務動態(tài)擴展ScheduledTaskRegistrar類示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-01-01
  • Java System類用法實戰(zhàn)案例

    Java System類用法實戰(zhàn)案例

    這篇文章主要介紹了Java System類用法,結合具體實例形式分析了java使用System類獲取系統(tǒng)環(huán)境變量信息相關操作技巧,需要的朋友可以參考下
    2019-07-07
  • Java中Base64加密解密舉例詳解

    Java中Base64加密解密舉例詳解

    Base64編碼是我們程序開發(fā)中經常使用到的編碼方法,它是一種基于用64個可打印字符來表示二進制數據的表示方法,這篇文章主要給大家介紹了關于Java中Base64加密解密的相關資料,需要的朋友可以參考下
    2024-05-05
  • Java按值傳遞和按址傳遞(面試常見)

    Java按值傳遞和按址傳遞(面試常見)

    這篇文章主要介紹了Java按值傳遞和按址傳遞(面試常見)知識,在面試筆試題中經常會遇到,今天小編給大家詳細介紹下,需要的朋友可以參考下
    2017-02-02
  • 詳解SpringBoot和Mybatis配置多數據源

    詳解SpringBoot和Mybatis配置多數據源

    本篇文章主要介紹了詳解SpringBoot和Mybatis配置多數據源,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-05-05
  • SpringBoot如何實現同域SSO(單點登錄)

    SpringBoot如何實現同域SSO(單點登錄)

    單點登錄(SingleSignOn,SSO),就是通過用戶的一次性鑒別登錄。即在多個應用系統(tǒng)中,只需要登錄一次,就可以訪問其他相互信任的應用系統(tǒng),本文將介紹SpringBoot如何實現同域SSO(單點登錄)
    2021-05-05
  • Nacos?動態(tài)服務發(fā)現、配置和服務管理平臺初體驗

    Nacos?動態(tài)服務發(fā)現、配置和服務管理平臺初體驗

    這篇文章主要介紹了Nacos?動態(tài)服務發(fā)現、配置和服務管理平臺初體驗的相關資料,需要的朋友可以參考下
    2022-09-09

最新評論