1. 程式人生 > >java.lang.NoClassDefFoundError和java.lang.ClassNotFoundException

java.lang.NoClassDefFoundError和java.lang.ClassNotFoundException

NoClassDefFoundError錯誤發生的原因

NoClassDefFoundError錯誤的發生,是因為Java虛擬機器在編譯時能找到合適的類,而在執行時不能找到合適的類導致的錯誤。例如在執行時我們想呼叫某個類的方法或者訪問這個類的靜態成員的時候,發現這個類不可用,此時Java虛擬機器就會丟擲NoClassDefFoundError錯誤。與ClassNotFoundException的不同在於,這個錯誤發生只在執行時需要載入對應的類不成功,而不是編譯時發生。很多Java開發者很容易在這裡把這兩個錯誤搞混。

簡單總結就是,NoClassDefFoundError發生在編譯時對應的類可用,而執行時在Java的classpath路徑中,對應的類不可用導致的錯誤。

NoClassDefFoundError和ClassNotFoundException區別

我們經常被java.lang.ClassNotFoundException和java.lang.NoClassDefFoundError這兩個錯誤迷惑不清,儘管他們都與Java classpath有關,但是他們完全不同。NoClassDefFoundError發生在JVM在動態執行時,根據你提供的類名,在classpath中找到對應的類進行載入,但當它找不到這個類時,就發生了java.lang.NoClassDefFoundError的錯誤,而ClassNotFoundException是在編譯的時候在classpath中找不到對應的類而發生的錯誤。ClassNotFoundException比NoClassDefFoundError容易解決,是因為在編譯時我們就知道錯誤發生,並且完全是由於環境的問題導致。而如果你在J2EE的環境下工作,並且得到NoClassDefFoundError的異常,而且對應的錯誤的類是確實存在的,這說明這個類對於類載入器來說,可能是不可見的。

怎麼解決NoClassDefFoundError錯誤

根據前文,很明顯NoClassDefFoundError的錯誤是因為在執行時類載入器在classpath下找不到需要載入的類,所以我們需要把對應的類載入到classpath中,或者檢查為什麼類在classpath中是不可用的,這個發生可能的原因如下:

  1. 對應的Class在java的classpath中不可用
  2. 你可能用jar命令執行你的程式,但類並沒有在jar檔案的manifest檔案中的classpath屬性中定義
  3. 可能程式的啟動指令碼覆蓋了原來的classpath環境變數
  4. 因為NoClassDefFoundError是java.lang.LinkageError的一個子類,所以可能由於程式依賴的原生的類庫不可用而導致
  5. 檢查日誌檔案中是否有java.lang.ExceptionInInitializerError這樣的錯誤,NoClassDefFoundError有可能是由於靜態初始化失敗導致的(這是我遇到的問題的解決辦法)
  6. 如果你工作在J2EE的環境,有多個不同的類載入器,也可能導致NoClassDefFoundError。
  7. NoClassDefFoundError也可能由於類的靜態初始化模組錯誤導致,當你的類執行一些靜態初始化模組操作,如果初始化模組丟擲異常,哪些依賴這個類的其他類會丟擲NoClassDefFoundError的錯誤。如果你檢視程式日誌,會發現一些java.lang.ExceptionInInitializerError的錯誤日誌,ExceptionInInitializerError的錯誤會導致java.lang.NoClassDefFoundError: Could not initialize class