Java:NoClassDefFoundError:org / json / JSONException

时间:2016-04-26 14:45:45

标签: java json jar

所以,我整整都花了这个问题 我确定,我正在使用正确的类路径 此外,我还有其他软件包作为依赖项,它们运行良好。

我有一个使用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

3 个答案:

答案 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-assemblyOneJar等工具可能就是您所需要的。

答案 2 :(得分:0)

所以,解决方案:

据我所知,访问jar中jar文件内容的唯一方法是编写自己的类加载器。
没有它,必须提取jar文件,并且必须将提取的内容包含在output.jar中。