所以,我整整都花了这个问题 我确定,我正在使用正确的类路径 此外,我还有其他软件包作为依赖项,它们运行良好。
我有一个使用org.json的课程。*
此类中还有一些其他外包装。
所有这些依赖都作为jar文件放在我的/ path / to / libs /中
json-20160212.jar就是其中之一。
我用
编译我的来源javac \
-cp "src/:/path/to/libs/json-20160212.jar:/path/to/libs/other.jar:/path/to/libs/another.jar" \
-d dst/ \
src/com/example/Source.java
编辑没有问题 然后,我从我的班级文件中创建jar 清单:
Main-Class: com.example.Source
Class-Path: /path/to/libs/json-20160212.jar
/path/to/libs/other.jar
/path/to/libs/another.jar
命令行:
jar cfm output.jar manifest -C dst/ ./com
我得到了这个清单的jar:
Manifest-Version: 1.0
Class-Path: /path/to/libs/json-20160212.jar /path/to/libs/other.jar /p
ath/to/libs/another.jar
Created-By: 1.7.0_101 (Oracle Corporation)
Main-Class: com.example.Source
正如我所理解的那样,编译清单可以分割线条。
现在,我从命令行运行我的应用程序并收到此错误:
Exception in thread "Thread-0" java.lang.NoClassDefFoundError: org/json/JSONException
at com.example.Source.run(Source.java:30)
Caused by: java.lang.ClassNotFoundException: org.json.JSONException
at java.net.URLClassLoader$1.run(URLClassLoader.java:366)
at java.net.URLClassLoader$1.run(URLClassLoader.java:355)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(URLClassLoader.java:354)
at java.lang.ClassLoader.loadClass(ClassLoader.java:425)
at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)
at java.lang.ClassLoader.loadClass(ClassLoader.java:358)
... 1 more
据我所知,这意味着,org.json.JSONException在编译时是正常的但在运行时丢失了。
但是我该如何处理这些信息呢?
我有那个档案。它在编译期间和运行时都处于适当位置
还有其他的依赖,他们的罐子也在那个地方。
如果我从我的应用程序中删除JSON使用情况,一切正常 所以,我可以得出结论,它是包org.json本身,这就产生了问题。
我该怎么做才能让它发挥作用?
更新
现在,我做了这些改变:
我的目录结构:
libs/
json-20160212.jar
other.jar
another.jar
src/
com/
example/
Source.java
dst/
清单:
Main-Class: com.example.Source
Class-Path: libs/json-20160212.jar
libs/other.jar
libs/another.jar
汇编:
javac \
-cp "src/:libs/json-20160212.jar:libs/other.jar:libs/another.jar" \
-d dst/ \
src/com/example/Source.java
Jarchiving:
jar cfm dst/output.jar manifest -C dst/ ./com ./libs
我得到的结构为例外的jar:
META-INF/
META-INF/MANIFEST.MF
com/
com/example/
com/example/Source.class
libs/
libs/json-20160212.jar
libs/other.jar
libs/another.jar
我正在使用java -jar dst/output.jar
运行它
结果相同:java.lang.NoClassDefFoundError: org/json/JSONException
答案 0 :(得分:1)
问题是您的运行时类路径。这个错误并没有什么神奇之处。它很简单意味着org.json.JSONException不在您的运行时类路径上。找到包含此类的jar并将其放在运行时类路径中。
请注意,运行时所需的jar /类不一定与编译所需的类相同。您的运行时类路径上经常需要比编译类路径更多的东西。如果在您正在编译的代码中没有显式使用JSONException,那么它就不必在您的编译类路径上。但是,如果您的代码的某个依赖项需要JSONException并且它不在您的运行时类路径上,那么您将获得NoClassDefFoundError。
可能发生的另一个问题是您的类路径上有2个不同版本的json jar。通常,类路径上的类的第一个版本被加载而另一个被忽略。如果第一个jar没有你需要的JSONException的版本/签名,但是第二个jar,你仍然会忽略正确的类,因为它在类路径上更进一步。
答案 1 :(得分:1)
问题似乎是您没有将相关的罐子添加到生成的罐子里。
我创建了一个类似的测试jar,具有以下结构(使用jar tf检查)...
META-INF /
META-INF / MANIFEST.MF
BeanIt.class
TestBean.class
LIB /
LIB / opencsv-3.7.jar
LIB /公地lang3-3.4.jar
我的清单......
Main-Class:BeanIt
类路径:lib / opencsv-3.7.jar lib / commons-lang3-3.4.jar
为了创建这个jar,你需要执行一个与此类似的命令......
jar cfm App.jar MANIFEST.MF BeanIt.class TestBean.class lib
您可以看到我已将我的lib文件夹添加到jar中,并在清单中的类路径中引用其内容。
因此,您可以更新现有的lib,就像这样......
jar uf App.jar path
其中path是路径/ / lib目录的根路径。它只会将它添加到你的罐子里。
您可以先使用jar tf
检查您的jar,看看它包含的内容。
如果你仍然难以开始工作,那么你可以看一下" FAT JAR"解决方案,您可以扩展所有内部jar类并将它们全部展平为包含所有必需类的单个JAR。他们使用决策机制来处理不同JAR中的类冲突。如果您无法按照预期的方式运行JAR,那么sbt-assembly或OneJar等工具可能就是您所需要的。
答案 2 :(得分:0)
所以,解决方案:
据我所知,访问jar中jar文件内容的唯一方法是编写自己的类加载器。
没有它,必须提取jar文件,并且必须将提取的内容包含在output.jar中。