26

我有一个测试:

public class ResourceTest {
    @Test
    public void test() throws ClassNotFoundException {
        Class.forName("javax.annotation.Resource");
    }
}

它试图访问javax.annotation.Resource. 在 java 8 中它可以工作,但在 java 9(我使用的是 Oracle JDK 9)中它失败并显示ClassNotFoundException. 正如Spring: @Resource injection 在 JDK9 下所解释的那样,javax.annotation.Resource默认情况下 Java 9 中的 JDK 不可用。

我正在尝试使用模块描述符访问它:

module test {
    requires java.xml.ws.annotation;
    requires junit;
}

在这里,我特别请求访问java.xml.ws.annotation模块(其中包含javax.annotation.Resource)。但测试仍然失败。

当我删除该requires子句并添加包含的依赖项(作为库)javax.annotations.Resource时,它可以工作:

    <dependency>
        <groupId>javax.annotation</groupId>
        <artifactId>javax.annotation-api</artifactId>
        <version>1.3.1</version>
    </dependency>

当我同时添加它们(pom.xml和Maven 依赖项requires java.xml.ws.annotation)时,IDEA 中的编译失败并显示以下消息:

the unnamed module reads package javax.annotation from both java.xml.ws.annotation and java.annotation

但是 Maven 构建仍然成功!

如果我java.xml.ws.annotation通过命令行添加模块,它可以工作(没有 Maven 依赖项和with requires子句):

mvn clean test -DargLine="--add-modules java.xml.ws.annotation"

我的模块描述有问题吗?如何在javax.annotation.Resource没有命令行开关的情况下访问 JDK 提供的访问权限?

测试项目位于https://github.com/rpuch/test-resource-jdk9

4

2 回答 2

40

只是为了清除这里的一些混乱。您在问题中陈述的工作方式是替代方案,不应像您已经看到的那样组合。

未命名的模块从 java.xml.ws.annotation 和 java.annotation 读取包 javax.annotation


所以它的工作方式是:

您可以使用编译器参数添加模块

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-compiler-plugin</artifactId>
    <version>3.7.0</version>
    <configuration>
        <release>9</release>
        <compilerArgs>
            <arg>--add-modules</arg>
            <arg>java.xml.ws.annotation</arg>
        </compilerArgs>
    </configuration>
</plugin>

或者

使用javax.xml.ws.annotation作为可升级模块,此时您可以使用依赖项

<dependency>
    <groupId>javax.annotation</groupId>
    <artifactId>javax.annotation-api</artifactId>
    <version>1.3.1</version>
</dependency>

理想情况下,这将是一个更可取的选择,因为前者只是使用标记为@Deprecated的模块的替代方案forRemoval


因此,required 子句本身不足以访问模块......对于所有 JDK 提供的模块(不包括 java.base)都是如此,还是仅对已弃用的模块如此?

不,这requires只是声明的一部分。[想一想,在 JDK 9 之前,如果您import some.foo.bar;在类中使用了未添加为库(类路径)的语句,那会起作用吗?]。标记为必需的模块必须位于模块路径上,您才能访问它。


更新- 使用 JDK/11 或更高版本将不再支持第一个选项,其中目标是删除 Java EE 和 CORBA 模块的 JEP 。

于 2017-09-30T11:08:59.243 回答
2

对于 gradle build,将以下内容添加到 build.gradle 作品中:

compile 'javax.annotation:jsr250-api:1.0'

tasks.withType(AbstractCompile) {
    options.compilerArgs += ["--add-modules", "java.xml.bind"]
}

tasks.withType(Test) {
    jvmArgs += ["--add-modules", "java.xml.bind"]
}
于 2018-04-14T08:12:01.687 回答