协慌网

登录 贡献 社区

如何解决 java.lang.NoClassDefFoundError:Java 9 中的 javax / xml / bind / JAXBException

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

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

答案

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

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

幸运的是,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. 您可以使用JDK_JAVA_OPTIONS环境变量设置任何仅 Java 9 + 的选项。 Java 9 + 版的java启动器会自动读取此环境变量。
  2. 您可以添加-XX:+IgnoreUnrecognizedVMOptions以使 JVM 静默忽略无法识别的选项,而不会-XX:+IgnoreUnrecognizedVMOptions 。但是要当心! JVM 将不再为您验证您使用的任何其他命令行参数。该选项与 Oracle / OpenJDK 以及 IBM JDK(自 JDK 8sr4 起)一起使用。

备用快速解决方案:(仅 JDK 9/10)

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


正确的长期解决方案:(JDK 9 和更高版本)

上面列出的 Java EE API 模块都标记为@Deprecated(forRemoval=true)因为它们计划Java 11 中 删除 。因此, --add-module方法将不再在现成的 Java 11 中起作用。

在 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:模块系统。

对于 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"
}

就我而言(春季启动胖子),我只是将以下内容添加到 pom.xml 中。

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

在最近的 JDK 9.0.1 中,这些解决方案都对我不起作用。

我发现,此依赖关系列表足以正常运行,因此您无需显式指定--add-module (尽管在这些依赖关系的 pom 内指定了它)。您唯一需要的是指定以下依赖项列表:

<dependencies>
    <dependency>
        <groupId>javax.xml.bind</groupId>
        <artifactId>jaxb-api</artifactId>
        <version>2.3.0</version>
    </dependency>
    <dependency>
        <groupId>com.sun.xml.bind</groupId>
        <artifactId>jaxb-impl</artifactId>
        <version>2.3.0</version>
    </dependency>
    <dependency>
        <groupId>org.glassfish.jaxb</groupId>
        <artifactId>jaxb-runtime</artifactId>
        <version>2.3.0</version>
    </dependency>
    <dependency>
        <groupId>javax.activation</groupId>
        <artifactId>activation</artifactId>
        <version>1.1.1</version>
    </dependency>
</dependencies>