怎么解决java.lang.NoClassDefFoundError错误
ClassNotfoundException时在编译时JVM加载不到类或者找不到类导致的;
而NoClassDefError是在运行时JVM加载不到类或者找不到类
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加载不到需要的类或加载了错误版本的类。
3.找不到jar包
如果是使用在lib下的包要设置这个两项为true,这个两项是在maven下面导入的
<configuration>
<executable>true</executable>
<includeSystemScope>true</includeSystemScope>
</configuration>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<executable>true</executable>
<includeSystemScope>true</includeSystemScope>
</configuration>
</plugin>
</plugins>
</build>