Exception in thread main java.lang.NoClassDefFoundError错误解决方法
错误描述
javachelloworld.java能够通过。但是javahelloworld出现错误:
hadoop@xuwei-erplab:~/jarfile$javaHelloWorld Exceptioninthread"main"java.lang.NoClassDefFoundError:HelloWorld(wrongname:org/xuwei/HelloWorld) atjava.lang.ClassLoader.defineClass1(NativeMethod) atjava.lang.ClassLoader.defineClassCond(ClassLoader.java:631) atjava.lang.ClassLoader.defineClass(ClassLoader.java:615) atjava.security.SecureClassLoader.defineClass(SecureClassLoader.java:141) atjava.net.URLClassLoader.defineClass(URLClassLoader.java:283) atjava.net.URLClassLoader.access$000(URLClassLoader.java:58) atjava.net.URLClassLoader$1.run(URLClassLoader.java:197) atjava.security.AccessController.doPrivileged(NativeMethod) atjava.net.URLClassLoader.findClass(URLClassLoader.java:190) atjava.lang.ClassLoader.loadClass(ClassLoader.java:306) atsun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:301) atjava.lang.ClassLoader.loadClass(ClassLoader.java:247) Couldnotfindthemainclass:HelloWorld.Programwillexit.
2、问题解决
在文献1中很多人提到都是因为环境变量classpath配置错误。说是没有加上当前路径"."。但是我查看了自己的classpath为
exportCLASSPATH=.:$JAVA_HOME/lib/tools.jar:$JAVA_HOME/lib/dt.jar:$CLASSPATH
这表明我的classpath配置是正确的。这个时候我找到了文献2,里面提到了helloword分为带包名和不带包名两类。
3、不含包层次的HelloWorld.java
publicclassHelloWorld{ publicstaticvoidmain(Stringargs[]){ System.out.println("HelloWorld!"); } }
保存在/home/hadoop/jarfile下,使用javac命令编译:
$javacHelloWorld.java
运行:
$javaHelloWorld
屏幕打印出:
HelloWorld!
4、初学者常犯的错误
4.1.运行时,带了.class后缀
如果你试图使用如下命令:(下面的命令都是假设在HelloWorld.java所在目录执行,即/home/hadoop/jarfile)
javaHelloWorld.class
系统会误认为你运行的是HelloWorld包下的名为class的类文件,会到系统的CLASSPATH下(一般都包括当前目录)企图寻找HelloWorld.class.class这样的类,这样的类当然不存在了;并且也不可能存在,因为class是关键字,不能作为一个类的名字。所以会报如下错误信息:
Exceptioninthread"main"java.lang.NoClassDefFoundError:HelloWorld/class
4.2.文件名大小写错误
对于像Windows这样的系统,编译时可以不关心大小写(linux区分大小写)。比如编译HelloWorld.java时,也可以使用:
javachelloworld.java
也可以编译通过,但产生的类文件仍然是和源文件相符的:HelloWorld.class。但在运行时一定要注意大小写,比如试图使用如下命令运行:
javahelloworld
将报类似于1中的错误:
Exceptioninthread"main"java.lang.NoClassDefFoundError:helloworld(wrongname:HelloWorld)
5、包含包层次的HelloWorld.java
比如上面的HelloWorld.java修改如下:
packageorg.myorg; publicclassHelloWorld{ publicstaticvoidmain(Stringargs[]){ System.out.println("HelloWorld!"); } }
编译时有两种方法
5.1.直接编译
javacHelloWorld.java
此时在当前目录下输出HelloWorld.class。此时,运行不能使用上面相同的方法,使用:
javaHelloWorld
运行时,出现如下错误:
Exceptioninthread"main"java.lang.NoClassDefFoundError:HelloWorld(wrongname:org/myorg/HelloWorld)
从上述错误信息你也可以看到,系统可以找到HelloWorld类(因为当前路径包含在CLASSPATH中,具体为什么会提示wrongname,有兴趣的朋友参见Java语言规范),但这个类属于org.myogr包。所以,你要做的就是按照上述包层次,相应的创建目录层次,把上面生成的HelloWorld.class放到/home/hadoop/jarfile/org/myorg目录下。HelloWorld.java在/home/hadoop/jarfile/目录下。运行:
javaorg.myorg.HelloWorld
系统打印出:
HelloWorld!
这儿要注意的是,不能使用javaorg\myorg\HelloWorld来运行,此时同样会出现如下错误:
Exceptioninthread"main"java.lang.NoClassDefFoundError:org\myorg\HelloWorld(wrongname:org\myorg\HelloWorld)
是不是有点怪怪的,那没办法。以后对Java的包有更深的认识时,就会明白了。
5.2.使用-d<directory>编译选项
是不是觉得上面的编译方法有点麻烦,能不能自动在当前路径(或任意指定的路径)下生成包层次呢?有!使用-d<directory>编译选项就能做到。
javac-d.HelloWorld.java
此时,在当前目录(/home/hadoop/jarfile)下就生成了一个org\myorg目录(/home/hadoop/jarfile/org/myorg),并且输出的.class文件也在里面。运行:
javaorg.myorg.HelloWorld
系统打印:
HelloWorld!
如果你想把生成的类文件集中存放在一个目录中,比如:/home/hadoop/jarfile/test下,那么你首先创建这个目录,然后编译时:
javac-d/home/hadoop/jarfile/testHelloWorld.java
就可以把生成的类文件放到/home/hadoop/jarfile/test目录下,并且按照包层次相应的创建目录路径。你可以在/home/hadoop/jarfile/test/org/myorg下找到HelloWorld.class文件。此时使用如下命令可以正确运行(注意如果要用到其它类,请在CLASSPATH中设好):
hadoop@xuwei-erplab:~/jarfile/test$javaorg.xuwei2.HelloWorld
注意上述命令是在/home/hadoop/jarfile/test下执行的。
如果不行可以参考下面的方法:
第二个问题解决方法:
这是因为JDK的版本而产生的问题,我装的是OpenJDK,会出现JAR包的缺失,导致启动报错,所以这里需要更换为Oracle官方给出的JDK
1、去Oracle官方下载一个这样的包:jdk-7u79-linux-x64.tar.gz
2、解压之后,移动到/usr/local/java目录下
3、添加环境变量:vim /etc/profile,文件最后添加下面几行
exportJAVA_HOME=/usr/local/java/jdk1.7.0_79//实际的JDK路径 exportCLASSPATH=.:$JAVA_HOME/jre/lib/rt.jar:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar exportPATH=$PATH:$JAVA_HOME/bin
4、完成上述安装之后,再启动Tomcat,发现还会报错:NeithertheJAVA_HOMEnortheJRE_HOMEenvironmentvariableisdefined
这个问题是因为Tomcat没有识别出JDK的环境变量
5、修改Tomcat中bin目录的catalina.sh文件,在文件的开头,加入下面代码:
exportJAVA_HOME=/usr/local/java/jdk1.7.0_79 ////实际的JDK路径 exportJRE_HOME=/usr/local/java/jdk1.7.0_79/jre
最后启动Tomcat,日志打印正常,浏览器也能够访问,问题解决
Exceptioninthread"main"java.lang.InternalError
atsun.security.ec.SunEC.initialize(NativeMethod)
atsun.security.ec.SunEC.access$000(SunEC.java:49)
atsun.security.ec.SunEC$1.run(SunEC.java:61)