ClassNotfoundException VS NoClassDefFoundError
ClassNotfoundException时在编译时JVM加载不到类或者找不到类导致的;
而NoClassDefError是在运行时JVM加载不到类或者找不到类
NoClassDefFoundError异常原因
1. 一种情况就是因为静态变量加载不到原因
NoClassDefFoundError错误产生的原因是:JVM在编译的时候能找到调用方法或静态变量所在的类,但在运行的时候找不到此类而引发的错误。如下面的例子:
public class TestNoClassDefFoundError {
public static void main(String[] args) throws InterruptedException {
TestNoClassDefFoundError sample = new TestNoClassDefFoundError();
sample.getClassWithInitErrors();
}
private void getClassWithInitErrors() throws InterruptedException {
System.out.println("第一次new");
Thread.sleep(500);
try {
//第一次new ClassWithInitErrors类,JVM会加载该类,初始化该类的静态变量或执行静态块
new ClassWithInitErrors();
} catch (Throwable t) {
//因为初始化静态变量失败,所以加载类失败。
t.printStackTrace();
}
Thread.sleep(500);
System.out.println("-----------------------------------------------------");
System.out.println("第二次new");
Thread.sleep(500);
try {
//第二次new ClassWithInitErrors类,JVM不会再加载该类,而是抛出NoClassDefFoundError异常
new ClassWithInitErrors();
} catch (Throwable t) {
t.printStackTrace();
}
Thread.sleep(500);
System.out.println("-----------------------------------------------------");
System.out.println("第三次new");
Thread.sleep(500);
try {
//第三次new ClassWithInitErrors类,JVM不会再加载该类,而是抛出NoClassDefFoundError异常
new ClassWithInitErrors();
} catch (Throwable t) {
t.printStackTrace();
}
}
}
class ClassWithInitErrors {
static int data = 1 / 0;
}
执行结果如下:
第一次new
java.lang.ExceptionInInitializerError
at TestNoClassDefFoundError.getClassWithInitErrors(TestNoClassDefFoundError.java:12)
at TestNoClassDefFoundError.main(TestNoClassDefFoundError.java:4)
Caused by: java.lang.ArithmeticException: / by zero
at ClassWithInitErrors.<clinit>(TestNoClassDefFoundError.java:42)
... 2 more
-----------------------------------------------------
第二次new
java.lang.NoClassDefFoundError: Could not initialize class ClassWithInitErrors
at TestNoClassDefFoundError.getClassWithInitErrors(TestNoClassDefFoundError.java:24)
at TestNoClassDefFoundError.main(TestNoClassDefFoundError.java:4)
-----------------------------------------------------
第三次new
java.lang.NoClassDefFoundError: Could not initialize class ClassWithInitErrors
at TestNoClassDefFoundError.getClassWithInitErrors(TestNoClassDefFoundError.java:34)
at TestNoClassDefFoundError.main(TestNoClassDefFoundError.java:4)
2. Jar包冲突
- java.lang.ClassNotFoundException:即java类找不到。这类典型异常通常是由于,没有在依赖管理中声明版本,maven的仲裁的时候选取了错误的版本,而这个版本缺少我们需要的某个class而导致该错误。例如httpclient-4.4.jar升级到httpclient-4.36.jar时,类org.apache.http.conn.ssl.NoopHostnameVerifier被去掉了,如果此时我们本来需要的是4.4版本,且用到了NoopHostnameVerifier这个类,而maven仲裁时选择了4.6,则会导致ClassNotFoundException异常。
- java.lang.NoSuchMethodError:即找不到特定方法,第一类冲突和第二类冲突都可能导致该问题——加载的类不正确。若是第一类冲突,则是由于错误版本的Jar包与所需要版本的Jar包中的类接口不一致导致,例如antlr-2.7.2.jar升级到antlr-2.7.6.Jar时,接口antlr.collections.AST.getLine()发生变动,当maven仲裁选择了错误版本而加载了错误版本的类AST,则会导致该异常;若是第二类冲突,则是由于不同Jar包含有的同名类接口不一致导致,典型的案例:Apache的commons-lang包,2.x升级到3.x时,包名直接从commons-lang改为commons-lang3,部分接口也有所改动,由于包名不同和传递性依赖,经常会出现两种Jar包同时在classpath下,org.apache.commons.lang.StringUtils.isBlank就是其中有差异的接口之一,由于Jar包的加载顺序,导致加载了错误版本的StringUtils类,就可能出现NoSuchMethodError异常。
- java.lang.NoClassDefFoundError:原因和上述雷同,就不作具体案例分析了。也就是同一个Jar包出现了多个不同版本,并选择了错误的版本而导致JVM加载不到需要的类或加载了错误版本的类。
如何解决NoClassDefFoundError
定位了冲突类的Jar包之后,通过mvn dependency:tree -Dverbose -Dincludes=<groupId>:<artifactId>
查看是哪些地方引入的Jar包的这个版本。
然后可用<excludes>
排除不需要的Jar包版本或者在依赖管理<dependencyManagement>
中申明版本可用<excludes>
排除不需要的Jar包版本或者在依赖管理<dependencyManagement>
中申明版本。