【问题标题】:Java: NoClassDefFoundError: org/json/JSONExceptionJava:NoClassDefFoundError:org/json/JSONException
【发布时间】:2016-04-26 14:45:44
【问题描述】:

所以,我花了一整天的时间来解决这个问题。
我确定,我使用的是正确的类路径。
另外,我还有其他包作为依赖项,它们运行良好。

我有一个使用 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 在编译时没问题,但在运行时丢失。
但是我必须如何处理这些信息?
我有那个文件。它在编译和运行时就位。
还有其他的依赖,他们的jar也在那个地方。

如果我从我的应用程序中删除 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

归档:

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

【问题讨论】:

  • 检查生成的 jar 中的内容。例如,JSON jar 实际上是在您的 jar 中的 /path/to/libs/json-20160212.jar 吗?
  • 我认为您没有在项目中添加特定的库。检查它并确保所有需要的 jar 都正确包含在项目中。
  • 按照上面的 jar 命令 (jar cfm output.jar manifest -C dst/ ./com),看起来您没有将库依赖项添加到 jar 中。包括 path/to/lib 以及您的类,之后它应该可以正常工作。

标签: java json jar


【解决方案1】:

问题在于您的运行时类路径。这个错误没有什么神奇之处。这很简单意味着 org.json.JSONException 不在您的运行时类路径中。找到其中包含此类的 jar,并将其放在您的运行时类路径中。

请注意,运行时所需的 jars / classes 不一定与编译所需的相同。您的运行时类路径通常比编译类路径需要更多。如果 JSONException 没有在您正在编译的代码中显式使用,那么它就不必在您的编译类路径中。但是,如果您的代码的依赖项之一需要 JSONException 并且它不在您的运行时类路径中,您将收到 NoClassDefFoundError。

另一个可能发生的问题是您的类路径中有 2 个不同版本的 json jar。通常类路径上的类的第一个版本被加载,而另一个被忽略。如果第一个 jar 中没有您需要的 JSONException 的版本/签名,但第二个有,那么您仍然会忽略正确的类,因为它位于类路径的更下方。

【讨论】:

    【解决方案2】:

    问题似乎是您没有将依赖 jar 添加到生成的 jar 中。

    我创建了一个类似的测试jar,结构如下(使用jar tf检查)...

    META-INF/
    META-INF/MANIFEST.MF
    BeanIt.class
    TestBean.class
    库/
    lib/opencsv-3.7.jar
    lib/commons-lang3-3.4.jar

    我的清单...

    主类: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 中,并在清单中的类路径中引用了它的内容。

    所以,您可以像这样更新现有的库...

    jar uf App.jar path
    

    其中 path 是 path/to/lib 目录的根路径。它只会将它添加到您的 jar 中。

    您可以先使用 jar tf 检查您的 jar,看看它包含什么。

    如果您仍然难以使其工作,那么您可以查看“FAT JAR”解决方案,通过该解决方案您可以扩展所有内部 jar 类并将它们全部展平为包含所有必要类的单个 JAR。他们使用决策机制来处理不同 JAR 中的类冲突。如果您无法让您的 JAR 以您期望的方式工作,那么您可能需要像 sbt-assemblyOneJar 这样的工具。

    【讨论】:

    • 我已经更新了原帖。请看一下。
    • 你的 json-20160212.jar 中是否确定了所需的类(org.json.JSONException)?还要仔细检查 JSONException 的大小写,以防它应该是 JsonException。只要该类在您的 json jar 中,json jar 在 jar 中,通过清单在类路径中引用并从您的调用类中正确引用,那么您应该很高兴。也许检查您是否也可以从其他 jar 中调用一个类。如果没问题,那么它可能是区分大小写的问题或缺课问题。
    【解决方案3】:

    所以,解决办法:

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

    【讨论】:

    • 你不应该这样做(正如我之前所说,我已经制作了一个包含另一个 jar 的 JAR,引用正确并且运行良好),但肯定有一些工具可以让你这样做例如 sbt-assembly(我在工作中使用这个)和 OneJar。他们使用决策机制来处理类冲突,并将所有类从依赖 jar 扩展到您的单个 fat jar。更新了我的答案,提供了一些关于胖罐解决方案的进一步建议。
    • 我认为,您能够运行您的 jar,因为您的 jar 中的文件结构与您的项目目录中的相同。所以你的库(dependences-jars)不是从jar中获取的,而是从你的本地文件系统中获取的。尝试将你的结果 jar 移动到独立的地方并运行它。
    • 另外,请查看官方文档:docs.oracle.com/javase/tutorial/deployment/jar/downman.html。第一个注释说:“Class-Path 标头指向本地网络上的类或 JAR 文件,而不是 JAR 文件中的 JAR 文件或可通过 Internet 协议访问的类。将 JAR 文件中的 JAR 文件中的类加载到类路径中,您必须编写自定义代码来加载这些类。”无论如何,非常感谢!你的回答给了我解决的方向。
    • 奇怪。尽管有文档,但它通常很有效。我想这取决于内部 JAR 里面的内容。如果它需要某种形式的嵌套类加载,那么它可能需要自定义处理,但对于简单的 JAR,它似乎没问题。您始终可以在 Eclipse 中使用 Export as Runnable Jar 选项。它为您完成所有这些工作并处理类加载方面。
    • 我使用 IntelliJ IDEA,但也有相同的选项。实际上,此选项提取 libs-jar 并将其内容打包到生成的 jar 中。这在本地开发的情况下非常方便。但就我而言,我不能依赖 IDE。我正在将源上传到远程服务器,并且在没有 IDE 的情况下在那里编译源。
    猜你喜欢
    • 2019-05-29
    • 2013-04-24
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-03-12
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多