如何解析java.lang.NoClassDefFoundError: javax/xml/bind/JAXBException

2022-08-31 03:57:26

我有一些使用JAXB API类的代码,这些类已作为Java 6/7/8中JDK的一部分提供。当我使用Java 9运行相同的代码时,在运行时我收到错误,指示找不到JAXB类。

自 Java 6 以来,JAXB 类已作为 JDK 的一部分提供,那么为什么 Java 9 不再能找到这些类呢?


答案 1

JAXB API 被认为是 Java EE API,因此不再包含在 Java SE 9 中的缺省类路径中。在Java 11中,它们从JDK中完全删除。

Java 9 引入了模块的概念,默认情况下,聚合模块在类路径(或者更确切地说,模块路径)上可用。顾名思义,聚合模块不包括传统上与 Java 6/7/8 捆绑在一起的 Java EE API。java.sejava.se

幸运的是,在 JDK 6/7/8 中提供的这些 Java EE API 仍然在 JDK 中,但默认情况下它们不在类路径上。以下模块中提供了额外的 Java EE API:

java.activation
java.corba
java.transaction
java.xml.bind  << This one contains the JAXB APIs
java.xml.ws
java.xml.ws.annotation

快速且肮脏的解决方案:(仅限 JDK 9/10)

要使 JAXB API 在运行时可用,请指定以下命令行选项:

--add-modules java.xml.bind

但是我仍然需要它来与Java 8一起使用!!!

如果您尝试使用较旧的 JDK 进行指定,它将爆炸,因为它是一个无法识别的选项。我建议以下两个选项之一:--add-modules

  1. 您可以使用环境变量设置任何仅限 Java 9+ 的选项。此环境变量由 Java 9+ 的启动器自动读取JDK_JAVA_OPTIONSjava
  2. 您可以添加 ,使 JVM 静默地忽略无法识别的选项,而不是爆炸。但要小心!JVM 将不再为您验证您使用的任何其他命令行参数。此选项适用于Oracle /OpenJDK以及IBM JDK(自JDK 8sr4起)。-XX:+IgnoreUnrecognizedVMOptions

替代快速解决方案:(仅限 JDK 9/10)

请注意,您可以通过指定该选项,使上述所有 Java EE 模块在运行时都可用。该模块是一个聚合模块,它包括上述 Java EE API 模块。请注意,这在 Java 11 上不起作用,因为在 Java 11 中已被删除。--add-modules java.se.eejava.se.eejava.se.eejava.se.ee


适当的长期解决方案:(JDK 9 及更高版本)

上面列出的 Java EE API 模块都已标记,因为它们在 Java 11 中被安排为要删除。因此,该方法将不再适用于开箱即用的Java 11。@Deprecated(forRemoval=true)--add-module

在 Java 11 和以后,您需要做的是在类路径或模块路径上包含您自己的 Java EE API 副本。例如,您可以将 JAX-B API 添加为 Maven 依赖项,如下所示:

<!-- API, java.xml.bind module -->
<dependency>
    <groupId>jakarta.xml.bind</groupId>
    <artifactId>jakarta.xml.bind-api</artifactId>
    <version>2.3.2</version>
</dependency>

<!-- Runtime, com.sun.xml.bind module -->
<dependency>
    <groupId>org.glassfish.jaxb</groupId>
    <artifactId>jaxb-runtime</artifactId>
    <version>2.3.2</version>
</dependency>

有关 JAXB 的更多详细信息,请参阅 JAXB 参考实现页面

有关 Java 模块化的完整详细信息,请参阅 JEP 261:模块系统

截至2022年7月,最新版本的bind-api和jaxb-runtime是4.0.0。所以你也可以使用

    <version>4.0.0</version>

...在这些依赖条款中。但是,如果这样做,包名称将从 更改为 。您需要修改源代码才能使用这些更高版本的 JAR。javax.xml.bind...jakarta.xml.bind...

对于 Gradle 或 Android Studio 开发者:(JDK 9 及更高版本)

将以下依赖项添加到文件中:build.gradle

dependencies {
    // JAX-B dependencies for JDK 9+
    implementation "jakarta.xml.bind:jakarta.xml.bind-api:2.3.2"
    implementation "org.glassfish.jaxb:jaxb-runtime:2.3.2"
}

答案 2

在我的情况下(弹簧靴脂肪罐),我只是将以下内容添加到pom.xml。

<dependency>
    <groupId>javax.xml.bind</groupId>
    <artifactId>jaxb-api</artifactId>
    <version>2.3.1</version>
</dependency>

推荐