Java中Error和Exception的異同以及運行時異常(Runtime exception)與檢查型異常(checked exception)的區別

一: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異常,它就是運行時異常,並且這種異常還是最常見的異常之一。