Java 9 模块系统是否支持可选的依赖项?

背景

在 maven 中,工件可以声明依赖关系

<optional>true</optional>

这意味着依赖项不是必需的,但如果存在,则可以使用。

模块系统的状态似乎指定模块只能读取它所需的模块。

问题

  • Java 9 模块系统是否确实不支持可选依赖项?
  • 为什么不呢?
  • Java 9 模块系统提供了哪些可选依赖项的替代方法?

用例

我有一个框架,它集成了应用程序可能使用或不使用的各种库。目前,此框架是一个单一的 JAR,它反映类路径以跳过缺少库的集成代码。

我想我们可以为每个配置将其拆分为一个单独的模块,但这会导致JAR数量的组合爆炸,因为我们不仅需要为每个可选依赖项使用单独的JAR,而且对于大多数可选依赖项对也需要一个单独的JAR...


答案 1

是的,支持可选依赖项。引用原始提案

扩展模块声明的语言,以允许在指令上使用修饰符,具有以下含义:staticrequires

  • 在编译时,表示强制依赖关系。如果在可观察模块中找不到合适的模块并得到解决,则会出现错误。requires static M

  • 在编译后的各个阶段中,表示可选依赖项。模块系统在分辨率期间不会在可观察模块中搜索合适的模块,但如果生成的模块图包含合适的模块,那么它将在执行通常的分辨率后健全性检查之前添加适当的可读性边缘。[...]requires static M

因此,一个假设的模块声明形式

module joda.beans {
    requires static joda.collect;
    ...
}

将确保模块在编译时可用,以便可以毫不费力地编译所引用的模块中的代码。但是,它不能保证在链接时或运行时可用。joda.collectjoda.beansjoda.collectjoda.collect

(与此同时,为该功能创建了官方文档

我为此写了一个演示。有趣的花絮是声明可选依赖项的模块...module-info.java

module org.codefx.demo.advent {
    // list the required modules
    requires org.codefx.demo.advent.calendar;
    // with 'static' the factories are only required at compile time;
    // to be present at run time either other modules most require them
    // or they must be added with the '--add-modules' command line option
    requires static org.codefx.demo.advent.factory.chocolate;
    requires static org.codefx.demo.advent.factory.quote;
}

...以及同一模块中想要从其可选依赖项访问类型的代码。它必须编写,以便在类型和/或不存在时优雅地失败:ChocolateFactoryQuoteFactory

private static List<SurpriseFactory> createSurpriseFactories() {
    return Stream.of(
            createChocolateFactoryIfAccessible(),
            createQuoteFactoryIfAccessible())
            .flatMap(Optional::stream)
            .collect(toList());
}

private static Optional<SurpriseFactory> createChocolateFactoryIfAccessible() {
    try {
        return Optional.of(new ChocolateFactory());
    } catch (NoClassDefFoundError er) {
        return Optional.empty();
    }
}

private static Optional<SurpriseFactory> createQuoteFactoryIfAccessible() {
    try {
        return Optional.of(new QuoteFactory());
    } catch (NoClassDefFoundError er) {
        return Optional.empty();
    }
}

最后,命令行可用于定义应用启动时使用的模块:

$java \
    --add-modules org.codefx.demo.advent.factory.chocolate,org.codefx.demo.advent.factory.quote \
    -p mods -m org.codefx.demo.advent

当然,其他模块也有可能非选择性地需要它们,这迫使JVM将它们包含在模块图中。


答案 2

推荐