解决运行jar包出错:ClassNotFoundException问题
作者:LuckyTHP
Java命令运行jar 报错
运行jar命令
java -jar xxx.jar
报错:
Caused by: java.lang.ClassNotFoundException:
原因:要运行的jar缺少依赖包
执行jar正确命令:
java -Xbootclasspath/a:/home/webuser/gogoal_platform/ggopenapi/lib/snakeyaml.jar -jar xxx.jar
解释:
-Xbootclasspath
:完全取代系统Java classpath.最好不用。-Xbootclasspath/a
: 在系统class加载后加载。一般用这个。-Xbootclasspath/p
: 在系统class加载前加载,注意使用,和系统类冲突就不好了.
win32 java -Xbootclasspath/a: some.jar;some2.jar; -jar test.jar unix java -Xbootclasspath/a: some.jar:some2.jar: -jar test.jar
win32系统每个jar用分号隔开,unix系统下用冒号隔开
方式二
AppClassloader来加载
这样就不需要classpath参数了
我们在MANIFEST.MF中添加如下代码:
Class-Path: lib/some.jar
lib是和test.jar同目录的一个子目录,test.jar要引用的some.jar包就在这里面。
(这里我刚开始理解成了把lib文件夹也打包进test.jar中,实践后没有成功,实际情况是lib和test.jar,是同一个目录一个层次级别的)然后测试运行,一切正常!
如果有多个jar包需要引用的情况:
Class-Path: lib/some.jar lib/some2.jar
每个单独的jar用空格隔开就可以了。注意使用相对路径。
另:如果META-INF
下包含INDEX.LIST文件的话,可能会使Class-Path配置失效。INDEX.LIST是Jar打包工具打包时生成的索引文件,删除对运行不产生影响。
jar包运行报错:Invalid or corrupt jarfile
执行:
java -jar xxx.jar
运行一个jar包时,提示报错:
Invalid or corrupt jarfile
该jar之前一直是正常运行的,这次只是增加了一项图片处理功能,引入了opencv-2413.jar。除了class文件的变化之外,唯一的修改就是MANIFEST.MF文件了。分析来看,问题很可能就出在这个文件上,将该文件还原,果然,jar至少能启动了。
不就是MANIFEST.MF文件中类路径新添加了一个opencv吗,为什么会这样呢?
MANIFEST.MF是一个非常脆弱的文件,格式要求比较严格,一个空格、一个回车符,都可能会导致该文件失效,从而使得对应的jar包非法:Invalid or corrupt jarfile!
MANIFEST.MF格式要求
1. 文件中的内容以key-value键值对的形式出现,key和value之间采用"英文冒号+空格"进行分隔,注意:冒号后的空格必须有;key必须顶格写,之前不能有空格。
2. 文件开头不能有空行,并且文件总是以Manifest-Version属性开头。
3. 文件必须以一个空行结束,注意:该空行不能有任何字符,包括空格。
4. 每行最长72个字符,如果超过的话,采用续行,换行继续必须以空格开头。
5. 对于Class-Path属性中的存在的路径,使用"/"分隔目录,与平台无关;多个jar包引用以空格分开。
针对MANIFEST.MF文件每行不能超过72个字符的说法,还有疑问。因为我的MANIFEST.MF文件中,Class-Path属性的很多行都是超过72个字符的,实际上并没有引起异常。
我的jar包之所以报错,就是违背了以上的第3条,文件虽然以空行结束,但该空行隐藏了一个空格,也就是上图中的第14行包含了一个空格。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。