欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Exception in thread main java.lang.NoClassDefFoundError错误解决方法

程序员文章站 2024-03-13 19:41:15
错误描述 javac helloworld.java能够通过。但是java helloworld出现错误: hadoop@xuwei-erplab:~/jarf...

错误描述
javac helloworld.java能够通过。但是java helloworld出现错误:

hadoop@xuwei-erplab:~/jarfile$ java helloworld 
exception in thread "main" java.lang.noclassdeffounderror: helloworld (wrong name: org/xuwei/helloworld)
	at java.lang.classloader.defineclass1(native method)
	at java.lang.classloader.defineclasscond(classloader.java:631)
	at java.lang.classloader.defineclass(classloader.java:615)
	at java.security.secureclassloader.defineclass(secureclassloader.java:141)
	at java.net.urlclassloader.defineclass(urlclassloader.java:283)
	at java.net.urlclassloader.access$000(urlclassloader.java:58)
	at java.net.urlclassloader$1.run(urlclassloader.java:197)
	at java.security.accesscontroller.doprivileged(native method)
	at java.net.urlclassloader.findclass(urlclassloader.java:190)
	at java.lang.classloader.loadclass(classloader.java:306)
	at sun.misc.launcher$appclassloader.loadclass(launcher.java:301)
	at java.lang.classloader.loadclass(classloader.java:247)
could not find the main class: helloworld. program will exit.

2、问题解决
在文献1中很多人提到都是因为环境变量classpath配置错误。说是没有加上当前路径"."。但是我查看了自己的classpath为

复制代码 代码如下:

export classpath=.:$java_home/lib/tools.jar:$java_home/lib/dt.jar:$classpath

这表明我的classpath配置是正确的。这个时候我找到了文献2,里面提到了helloword分为带包名和不带包名两类。
3、不含包层次的helloworld.java

public class helloworld {
	public static void main(string args[]){
		system.out.println("hello world!");
	}
}

保存在/home/hadoop/jarfile下,使用javac命令编译:

复制代码 代码如下:

$ javac helloworld.java  

运行:
复制代码 代码如下:

$ java helloworld 

屏幕打印出:
  hello world!

4、初学者常犯的错误
4.1. 运行时,带了.class后缀
        如果你试图使用如下命令:(下面的命令都是假设在helloworld.java所在目录执行,即/home/hadoop/jarfile)

复制代码 代码如下:

java helloworld.class 

系统会误认为你运行的是helloworld包下的名为class的类文件,会到系统的classpath下(一般都包括当前目录)企图寻找 helloworld.class.class这样的类,这样的类当然不存在了;并且也不可能存在,因为class是关键字,不能作为一个类的名字。所以会报如下错误信息:
  exception in thread "main" java.lang.noclassdeffounderror: helloworld/class
4.2. 文件名大小写错误
  对于像windows这样的系统,编译时可以不关心大小写(linux区分大小写)。比如编译helloworld.java时,也可以使用:

复制代码 代码如下:

javac helloworld.java

也可以编译通过,但产生的类文件仍然是和源文件相符的:helloworld.class。但在运行时一定要注意大小写,比如试图使用如下命令运行:

复制代码 代码如下:

java helloworld 

将报类似于1中的错误:
  exception in thread "main" java.lang.noclassdeffounderror: helloworld (wrong name: helloworld)
5、包含包层次的helloworld.java
  比如上面的helloworld.java修改如下:

package org.myorg;

public class helloworld {
	public static void main(string args[]){
		system.out.println("hello world!");
	}
}

编译时有两种方法
5.1. 直接编译

复制代码 代码如下:

javac helloworld.java 

  此时在当前目录下输出helloworld.class。此时,运行不能使用上面相同的方法,使用:

复制代码 代码如下:

java helloworld 

  运行时,出现如下错误:
  exception in thread "main" java.lang.noclassdeffounderror: helloworld (wrong name: org/myorg/helloworld)
  从上述错误信息你也可以看到,系统可以找到helloworld类(因为当前路径包含在classpath中,具体为什么会提示wrong name,有兴趣的朋友参见java语言规范),但这个类属于org.myogr包。所以,你要做的就是按照上述包层次,相应的创建目录层次,把上面生成的helloworld.class放到/home/hadoop/jarfile/org/myorg目录下。helloworld.java在/home/hadoop/jarfile/目录下。运行: 

复制代码 代码如下:

java org.myorg.helloworld 

系统打印出:
 hello world! 

这儿要注意的是,不能使用java org\myorg\helloworld来运行,此时同样会出现如下错误:
exception in thread "main" java.lang.noclassdeffounderror :org\myorg\helloworld (wrong name: 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文件也在里面。运行:

复制代码 代码如下:

java org.myorg.helloworld 

系统打印:

hello world!

  如果你想把生成的类文件集中存放在一个目录中,比如:/home/hadoop/jarfile/test下,那么你首先创建这个目录,然后编译时:

复制代码 代码如下:

javac -d /home/hadoop/jarfile/test helloworld.java 

  就可以把生成的类文件放到/home/hadoop/jarfile/test目录下,并且按照包层次相应的创建目录路径。你可以在/home/hadoop/jarfile/test/org/myorg下找到helloworld.class文件。此时使用如下命令可以正确运行(注意如果要用到其它类,请在classpath中设好):

复制代码 代码如下:

hadoop@xuwei-erplab:~/jarfile/test$ java org.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,文件最后添加下面几行

复制代码 代码如下:

export java_home=/usr/local/java/jdk1.7.0_79   //实际的jdk路径
export classpath=.:$java_home/jre/lib/rt.jar:$java_home/lib/dt.jar:$java_home/lib/tools.jar
export path=$path:$java_home/bin

4、完成上述安装之后,再启动tomcat,发现还会报错:neither the java_home nor the jre_home environment variable is defined

这个问题是因为tomcat没有识别出jdk的环境变量

5、修改tomcat中bin目录的catalina.sh文件,在文件的开头,加入下面代码:

复制代码 代码如下:

export java_home=/usr/local/java/jdk1.7.0_79  ////实际的jdk路径
export jre_home=/usr/local/java/jdk1.7.0_79/jre

最后启动tomcat,日志打印正常,浏览器也能够访问,问题解决

exception in thread "main" java.lang.internalerror
        at sun.security.ec.sunec.initialize(native method)
        at sun.security.ec.sunec.access$000(sunec.java:49)
        at sun.security.ec.sunec$1.run(sunec.java:61)