Java中Error和Exception的異同以及運行時異常(Runtime exception)與檢查型異常(checked exception)的區別
- 2020 年 5 月 13 日
- 筆記
- JAVA, 檢查型異常(CheckedException), 檢查型異常和運行時異常的異同, 檢查型異常和非檢查型異常的區別, 運行時異常(RuntimeException), 非檢查型異常(UncheckedException), 非運行時異常
一:Error和Exception的基本概念:
首先Exception和Error都是繼承於Throwable 類,在 Java 中只有 Throwable 類型的實例才可以被拋出(throw)或者捕獲(catch),它是異常處理機制的基本組成類型。
Exception 和 Error 體現了 Java 平台設計者對不同異常情況的分類,Exception和Error體現了JAVA這門語言對於異常處理的兩種方式。
Exception 是程序正常運行過程中可以預料到的意外情況,並且應該被開發者捕獲,進行相應的處理。
Error是java程序運行中不可預料的異常情況(正常情況下不大可能出現的情況),這種異常發生以後,會直接導致JVM不可處理或者不可恢復的情況。所以這種異常不可能抓取到,比如OutOfMemoryError、NoClassDefFoundError等。【表示由JVM所偵測到的無法預期的錯誤,由於這是屬於JVM層次的嚴重錯誤 ,導致JVM無法繼續執行,因此,這是不可捕捉到的,無法採取任何恢復的操作,頂多只能顯示錯誤信息。 Error類體系描述了Java運行系統中的內部錯誤以及資源耗盡的情形.應用程序不應該拋出這種類型的對象(一般是由虛擬機拋出).假如出現這種錯誤,除了儘力使程序安全退出外,在其他方面是無能為力的。】
其中的Exception又分為檢查性異常(checked)和非檢查性異常(unchecked)。兩個根本的區別在於,檢查性異常 必須在編寫代碼時,使用try catch捕獲(比如:IOException異常)。非檢查性異常 在代碼編寫使,可以忽略捕獲操作(比如:ArrayIndexOutOfBoundsException),這種異常是在代碼編寫或者使用過程中通過規範可以避免發生的,具體根據需要來判斷是否需要捕獲,並不會在編譯器強制要求。
切記,Error是Throw不是Exception 。
二者的不同之處(小總結):
Exception:
1.可以是可被控制(checked) 或不可控制的(unchecked)。
2.表示一個由程序員導致的錯誤。
3.應該在應用程序級被處理。
Error:
1.總是不可控制的(unchecked)。
2.經常用來用於表示系統錯誤或低層資源的錯誤。
3.如何可能的話,應該在系統級被捕捉。
如下是常見的 Error 和 Exception:
1)運行時異常(RuntimeException)也稱 【非檢查型異常 UncheckedException】:
- NullPropagation:空指針異常;
- ClassCastException:類型強制轉換異常
- IllegalArgumentException:傳遞非法參數異常
- IndexOutOfBoundsException:下標越界異常
- NumberFormatException:數字格式異常
- ArrayIndexOutOfBoundsException: 數組越界異常
- ArrayStoreException: 數據存儲異常,操作數組時類型不一致
- AruthmeticException: (算術異常)
- BufferOverflowException: (緩衝區溢出異常)
2)非運行時異常(CheckedException)也稱 【檢查型異常】:
- ClassNotFoundException:找不到指定 class 的異常
- IOException:IO 操作異常
- FileNotFoundException:文件不存在異常
- SQLException:SQL語句異常
- InterruptedException: (中斷異常-調用線程睡眠時候)
3)錯誤(Error):
- NoClassDefFoundError:找不到 class 定義異常
- StackOverflowError:深遞歸導致棧被耗盡而拋出的異常
- OutOfMemoryError:內存溢出異常
eg(代碼示例01)
下面代碼會導致 Java 堆棧溢出錯誤。
/ 通過無限遞歸演示堆棧溢出錯誤 class StackOverflow { public static void test(int i) { if (i == 0) { return; } else { test(i++); } } } public class ErrorEg { public static void main(String[] args) { // 執行StackOverflow方法 StackOverflow.test(5); } }
運行輸出為:
Exception in thread "main" java.lang.StackOverflowError at ch11.StackOverflow.test(ErrorEg.java:9) at ch11.StackOverflow.test(ErrorEg.java:9) at ch11.StackOverflow.test(ErrorEg.java:9) at ch11.StackOverflow.test(ErrorEg.java:9)
上面代碼通過無限遞歸調用最終引發了 java.lang.StackOverflowError 錯誤。
二:Error和Exception的使用場景:
eg01:經典的面試題目: 就是 NoClassDefFoundError 和 ClassNotFoundException 有什麼區別
區別一: NoClassDefFoundError它是Error,
ClassNotFoundException是Exception。 區別二:NoClassDefFoundError是JVM運行時通過classpath加載類時,找不到對應的類而拋出的錯誤。
ClassNotFoundException是在編譯過程中如果可能出現此異常,在編譯過程中必須將ClassNotFoundException異常拋出! NoClassDefFoundError發生場景如下: 1、類依賴的class或者jar不存在 (簡單說就是maven生成運行包後被篡改) 2、類文件存在,但是存在不同的域中 (簡單說就是引入的類不在對應的包下) 3、大小寫問題,javac編譯的時候是無視大小的,很有可能你編譯出來的class文件就與想要的不一樣!這個沒有做驗證 ClassNotFoundException發生場景如下: 1、調用class的forName方法時,找不到指定的類 2、ClassLoader 中的 findSystemClass() 方法時,找不到指定的類 舉例說明如下: Class.forName("abc"); 比如abc這個類不存項目中,代碼編寫時,就會提示此異常是檢查性異常,比如將此異常拋出。
三:運行時異常(Runtime exception)與檢查型異常(checked exception)的區別:
Java提供了兩類主要的異常:runtime exception和checked exception。
1:checked (檢查型異常)也就是我們經常遇到的IO異常,以及SQL異常都是這種異常。編譯器要檢查這類異常,檢查的目的一方面是因為該類異常的發生難以避免,
另一方面就是讓開發者去解決掉這類異常,所以稱為必須處理(try ...catch)的異常。如果不處理這類異常,集成開發環境中的編譯器一般會給出錯誤提示。
注意:如果一個方法沒有捕獲一個檢查性異常,那麼該方法必須使用 throws 關鍵字來聲明。throws 關鍵字放在方法名的後邊,eg: 方法名() throws IOExecption。
2:runtime exception(運行時異常),編譯器不會檢查這類異常,不檢查的則開發者在代碼的編輯編譯階段就不是必須處理,這類異常一般可以避免,因此無需處理(try ...catch),
如果不處理這類異常,集成開發環境中的編譯器也不會給出錯誤提示。 當出現這樣的異常時,總是由虛擬機接管。
eg:我們從來沒有人去處理過NullPointerException異常,它就是運行時異常,並且這種異常還是最常見的異常之一。