问题描述
java.lang.NoClassDefFoundError
,意味着找不到类定义,当JVM
或者ClassLoader
实例尝试装载该类的定义(这通常是一个方法调用或者new
表达式创建一个实例过程的一部分)而这个类定义并没有找时所抛出的错误。
问题原因
该异常一般出现在编译环境和运行环境不一致的情况下,就是说有可能在编译过后更改了Classpath
或者jar
包所以导致在运行的过程中JVM
或者ClassLoader
无法找到这个类的定义。
NoClassDefFoundError
错误的发生,是因为Java
虚拟机在编译时能找到合适的类,而在运行时不能找到合适的类导致的错误。例如在运行时我们想调用某个类的方法或者访问这个类的静态成员的时候,发现这个类不可用,此时Java
虚拟机就会抛出NoClassDefFoundError
错误。与ClassNotFoundException
的不同在于,这个错误发生只在运行时需要加载对应的类不成功,而不是编译时发生。很多Java
开发者很容易在这里把这两个错误搞混。
NoClassDefFoundError和ClassNotFoundException区别
我们经常被java.lang.ClassNotFoundException
和java.lang.NoClassDefFoundError
这两个错误迷惑不清,尽管他们都与Java classpath
有关,但是他们完全不同。NoClassDefFoundError
发生在JVM
在动态运行时,根据你提供的类名,在classpath
中找到对应的类进行加载,但当它找不到这个类时,就发生了java.lang.NoClassDefFoundError
的错误,而ClassNotFoundException
是在编译的时候在classpath
中找不到对应的类而发生的错误。ClassNotFoundException
比NoClassDefFoundError
容易解决,是因为在编译时我们就知道错误发生,并且完全是由于环境的问题导致。而如果你在J2EE
的环境下工作,并且得到NoClassDefFoundError
的异常,而且对应的错误的类是确实存在的,这说明这个类对于类加载器来说,可能是不可见的。
怎么解决NoClassDefFoundError错误
可能的原因如下:
- 对应的
Class
在java
的classpath
中不可用 - 你可能用
jar
命令运行你的程序,但类并没有在jar
文件的manifest
文件中的classpath
属性中定义 - 可能程序的启动脚本覆盖了原来的
classpath
环境变量 - 因为
NoClassDefFoundError
是java.lang.LinkageError
的一个子类,所以可能由于程序依赖的原生的类库不可用而导致 - 检查日志文件中是否有
java.lang.ExceptionInInitializerError
这样的错误,NoClassDefFoundError
有可能是由于静态初始化失败导致的 - 如果你工作在
J2EE
的环境,有多个不同的类加载器,也可能导致NoClassDefFoundError
- 跨进程调用 导致找不到那个类
NoClassDefFoundError解决示例
-
当发生由于缺少
jar
文件,或者jar
文件没有添加到classpath
,或者jar
的文件名发生变更会导致java.lang.NoClassDefFoundError
的错误。 -
当类不在
classpath
中时,这种情况很难确切的知道,但如果在程序中打印出System.getproperty(“java.classpath”)
,可以得到程序实际运行的classpath
-
运行时明确指定你认为程序能正常运行的
-classpath
参数,如果增加之后程序能正常运行,说明原来程序的classpath
被其他人覆盖了。 -
NoClassDefFoundError
也可能由于类的静态初始化模块错误导致,当你的类执行一些静态初始化模块操作,如果初始化模块抛出异常,哪些依赖这个类的其他类会抛出NoClassDefFoundError
的错误。如果你查看程序日志,会发现一些java.lang.ExceptionInInitializerError
的错误日志,ExceptionInInitializerError
的错误会导致java.lang.NoClassDefFoundError: Could not initialize class
,如下面的代码示例:
/**
* Java program to demonstrate how failure of static initialization subsequently cause
* java.lang.NoClassDefFoundError in Java.
* @author Javin Paul
*/
public class NoClassDefFoundErrorDueToStaticInitFailure {
public static void main(String args[]){
List<User> users = new ArrayList<User>(2);
for(int i=0; i<2; i++){
try{
users.add(new User(String.valueOf(i))); //will throw NoClassDefFoundError
}catch(Throwable t){
t.printStackTrace();
}
}
}
}
class User{
private static String USER_ID = getUserId();
public User(String id){
this.USER_ID = id;
}
private static String getUserId() {
throw new RuntimeException("UserId Not found");
}
}
Output
java.lang.ExceptionInInitializerError
at testing.NoClassDefFoundErrorDueToStaticInitFailure.main(NoClassDefFoundErrorDueToStaticInitFailure.java:23)
Caused by: java.lang.RuntimeException: UserId Not found
at testing.User.getUserId(NoClassDefFoundErrorDueToStaticInitFailure.java:41)
at testing.User.<clinit>(NoClassDefFoundErrorDueToStaticInitFailure.java:35)
... 1 more
java.lang.NoClassDefFoundError: Could not initialize class testing.User
at testing.NoClassDefFoundErrorDueToStaticInitFailure.main(NoClassDefFoundErrorDueToStaticInitFailure.java:23)
Read more: http://javarevisited.blogspot.com/2011/06/noclassdeffounderror-exception-in.html#ixzz3dqtbvHDy
-
由于
NoClassDefFoundError
是LinkageError
的子类,而LinkageError
的错误在依赖其他的类时会发生,所以如果你的程序依赖原生的类库和需要的dll
不存在时,有可能出现java.lang.NoClassDefFoundError
。这种错误也可能抛出java.lang.UnsatisfiedLinkError: no dll in java.library.path Exception Java
这样的异常。解决的办法是把依赖的类库和dll
跟你的jar
包放在一起。 -
如果你使用
Ant
构建脚本来生成jar
文件和manifest
文件,要确保Ant
脚本获取的是正确的classpath
值写入到manifest.mf
文件 -
Jar
文件的权限问题也可能导致NoClassDefFoundError
,如果你的程序运行在像linux
这样多用户的操作系统种,你需要把你应用相关的资源文件,如Jar
文件,类库文件,配置文件的权限单独分配给程序所属用户组,如果你使用了多个用户不同程序共享的jar
包时,很容易出现权限问题。比如其他用户应用所属权限的jar
包你的程序没有权限访问,会导致java.lang.NoClassDefFoundError
的错误。 -
基于
XML
配置的程序也可能导致NoClassDefFoundError
的错误。比如大多数Java
的框架像Spring
,Struts
使用xml
配置获取对应的bean
信息,如果你输入了错误的名称,程序可能会加载其他错误的类而导致NoClassDefFoundError
异常。我们在使用Spring MVC
框架或者Apache Struts
框架,在部署War
文件或者EAR
文件时就经常会出现Exception in thread “main” java.lang.NoClassDefFoundError。
-
在有多个
ClassLoader
的J2EE
的环境中,很容易出现NoClassDefFoundError
的错误。由于J2EE
没有指明标准的类加载器,使用的类加载器依赖与不同的容器像Tomcat
、WebLogic
,WebSphere
加载J2EE
的不同组件如War
包或者EJB-JAR
包。关于类加载器的相关知识可以参考这篇文章类加载器的工作原理。 -
总结来说,类加载器基于三个机制:委托、可见性和单一性,委托机制是指将加载一个类的请求交给父类加载器,如果这个父类加载器不能够找到或者加载这个类,那么再加载它。可见性的原理是子类的加载器可以看见所有的父类加载器加载的类,而父类加载器看不到子类加载器加载的类。单一性原理是指仅加载一个类一次,这是由委托机制确保子类加载器不会再次加载父类加载器加载过的类。现在假设一个
User
类在WAR
文件和EJB-JAR
文件都存在,并且被WAR ClassLoader
加载,而WAR ClassLoader
是加载EJB-JAR ClassLoader
的子ClassLoader
。当EJB-JAR
中代码引用这个User
类时,加载EJB-JAR
所有class
的Classloader
找不到这个类,因为这个类已经被EJB-JAR classloader
的子加载器WAR classloader
加载。 -
这会导致的结果就是对
User
类出现NoClassDefFoundError
异常,而如果在两个JAR
包中这个User
类都存在,如果你使用equals
方法比较两个类的对象时,会出现ClassCastException
的异常,因为两个不同类加载器加载的类无法进行比较。 -
有时候会出现
Exception in thread “main” java.lang.NoClassDefFoundError: com/sun/tools/javac/Main
这样的错误,这个错误说明你的Classpath
,PATH
或者JAVA_HOME
没有安装配置正确或者JDK
的安装不正确。这个问题的解决办法时重新安装你的JDK
。 -
Java
在执行linking
操作的时候,也可能导致NoClassDefFoundError
。例如在前面的脚本中,如果在编译完成之后,我们删除User
的编译文件,再运行程序,这个时候你就会直接得到NoClassDefFoundError
,而错误的消息只打印出User
类的名称。
java.lang.NoClassDefFoundError: testing/User
at testing.NoClassDefFoundErrorDueToStaticInitFailure.main(NoClassDefFoundErrorDueToStaticInitFailure.java:23)
实际情况
更新服务在停止旧服务之前没有休眠,而是立即启动了新服务。(测试休眠一会再启动新服务后不报错,所以猜测是这个原因。)