jar包执行报ClassNotFoundException

使用Eclipse打包jar包,指定了main class。

java -jar mongoCluster.jar 

但是运行的时候报ClassNotFoundException NoClassDefFoundError

 

经查找,由Eclipse自动打包生成MANIFEST.MF不正确,正确的MANIFEST.MF如下  

Manifest-Version: 1.0

Main-Class: net.java2000.test.jar.TestJar

Class-Path: spring.jar 

 lib/commons-logging-1.1.jar

 

这里特别说明一下 

1)在 Class-Path: 后面有一个空格,切记 

2)在 Class-Path: 后面写上你的jar 用空格分开 

3)MANIFEST有严格的长度限制,如果class-path长了,就必须换行,

切记在上一行末尾一定要有一个空格,下一行的开头一定要有2个空格 

4)最后一定要空2行,否则Eclipse打包时有可能把你的Class-Path 给忽略掉

 

实际上,上面的方法能在某些环境下起到一定作用,但是无法根除。

笔者试过在Windows上面可以找到jar,在Linux上面找不到jar的恶心情况。

stackoverflow上面有人说,其实是不可以把第三方jar打到自己的jar包里面。也不建议这么搞。

 

 

百度有如下的解决方法:

1)  使用 java -cp 来设置 classpath 对于 jar来说是无效的,因为根据jar的安全规定,其内部的Class-Path 会起作用,外部的会被屏蔽掉(注意是屏蔽掉,不是覆盖掉) 

 

2) Java自身提供了一个设置classpath的方案,那就是使用命令行参数 

-Xbootclasspath:      完全取代基本核心的Java class 搜索路径.  

                                    不常用,否则要重新写所有Java 核心class  

-Xbootclasspath/a: 后缀在核心class搜索路径后面.常用!!  

-Xbootclasspath/p: 前缀在核心class搜索路径前面.不常用,避免  

                                    引起不必要的冲突. 

语法如下:  

(分隔符与classpath参数类似,unix使用:号,windows使用;)  

java -Xbootclasspath/a:spring.jar;lib/commons-logging-1.1.jar -jar MyProject.jar 

 

3)当然,你把jar放到 {Java_home}/jre/lib/ext 这个目录下面也是可以的,应为JVM肯定会搜索这个目录。

 
注意,如果要用Eclipse编译jar包,就最好使用export runnable jar file。
使用export jar file会导致ClassNotFoundException的异常或NoClassDefFoundError。
 
最后,本人测试过是可以使用fat jar进行one-jar的打包方式。
其原理就是通过实现一个自定义的classloader来把所有的依赖包,打成一个jar包。
Eclipse上面有插件,但是0.0.31的插件只能支持到Eclipse Indigo版本。
 
最后实验得出
在MANIFEST.MF里面配置的class-path如下:
Class-Path: ./lib/XXX-1.3.jar 
在widows下面,会寻找jar包下的jar包。
/my.jar
  |--XXX-1.3.jar
而在Linux下面,则会按照与jar包处在相同的目录下寻找
/my.jar
/lib
  |--XXX-1.3jar
 
结论是,class-path是有用的,但是不同环境对它的解释也不一样。请慎用。
refer: http://blog.csdn.net/java2000_net/article/details/2522408
http://stackoverflow.com/questions/183292/classpath-including-jar-within-a-jar 

你可能感兴趣的:(Java)