32

随着 Java 9 的临近,我认为将我的一些项目移植到 Java 9 将是一个很好的学习练习。在我的一个项目中,我依赖于rxjavarxjavafx

dependencies {
    compile 'io.reactivex:rxjava:1.2.6'
    compile 'io.reactivex:rxjavafx:1.0.0'
    ...
}

我想将此项目创建为命名模块。为此,我需要创建一个module-info.java文件,并且需要在此处指定和的rxjava要求rxjavafx。但是,这些库还没有任何模块信息。

为了解决这个问题,我读到我需要创建 Automatic Modules。据我了解,我需要将 jar 重命名为一个简单的名称,rxjava然后在参数rxjavafx中列出这些 jar 。然后我在我的 jar 名称中--module-path添加一个requires指令。module-info.java

module com.foo.bar {
    requires rxjavafx;
    requires rxjava;
}

我编写了一个 gradle 任务来为我编辑 jar 名称,它似乎在大多数情况下都有效。它需要所有需要编译的 jars 并将它们重命名为不包含版本信息或斜杠。然后将文件连接成一个:单独的字符串:

tasks.withType(JavaCompile) {
    delete { delete '/tmp/gradle' }
    copy {
        from configurations.compile + configurations.testCompile
        into '/tmp/gradle'
        rename '(.*)-[0-9]+\\..*.jar', '$1.jar'
        rename { String fileName -> fileName.replace("-", "") }
    }
    options.compilerArgs += ['--module-path', fileTree(dir: '/tmp/gradle', include: '*.jar').getFiles().join(':')]
}

自然地,这些rx库共享它们的一些包名称......但这会导致编译器吐回错误,例如:

error: module  reads package rx.subscriptions from both rxjava and rxjavafx
error: module  reads package rx.schedulers from both rxjava and rxjavafx
error: module  reads package rx.observables from both rxjava and rxjavafx
error: module rxjava reads package rx.subscriptions from both rxjavafx and rxjava
error: module rxjava reads package rx.schedulers from both rxjavafx and rxjava
error: module rxjava reads package rx.observables from both rxjavafx and rxjava
error: module rxjavafx reads package rx.subscriptions from both rxjava and rxjavafx
error: module rxjavafx reads package rx.schedulers from both rxjava and rxjavafx
error: module rxjavafx reads package rx.observables from both rxjava and rxjavafx

似乎解决此问题的唯一方法是将其内容重新打包到一个 jar 中rxjava并将rxjavafx其添加为单个模块。虽然这似乎不是一个好的解决方案......

所以我的问题是:

  • 我是否正确使用了新的模块系统?
  • 我能做些什么来处理这个错误?和
  • 这些依赖项是否阻止我更新,或者我应该等待 rx 更新他们的库?

注意:我尝试使用标准java/运行它javac,它们会导致相同的问题。这也是我的java版本:

java version "9-ea"
Java(TM) SE Runtime Environment (build 9-ea+140)
Java HotSpot(TM) 64-Bit Server VM (build 9-ea+140, mixed mode)
4

3 回答 3

27

我是否正确使用了新的模块系统?

是的。您看到的是预期行为,这是因为 JPMS 模块不允许拆分包。

如果您不熟悉术语“拆分包”,它本质上是指同一包的两个成员来自两个不同的模块。

例如:
com.foo.A(来自 moduleA.jar)
com.foo.B(来自 moduleB.jar)

我能做些什么来处理这个错误?

你有两个选择:

  1. (更难)“未拆分”包依赖项。但是,如果您不熟悉库的内部运作,这可能会很困难或不可能
  2. (更容易)如上所述将两个罐子组合成一个罐子(因此是一个自动模块)。我同意这不是一个“好”的解决方案,但首先拆分包通常也不是一个好主意。

这些依赖项是否阻止我更新,或者我应该等待 rx 更新他们的库?

希望 rx 最终会在将来的某个时候更新他们的库以不具有拆分包。在那之前,我的建议是将两个罐子一起粉碎成一个罐子(选项#2)。

于 2017-02-21T03:38:58.130 回答
5

我有类似的问题:

error: module flyway.core reads package javax.transaction.xa from both jboss.transaction.api.1.2.spec and java.sql
error: module slf4j.api reads package javax.transaction.xa from both jboss.transaction.api.1.2.spec and java.sql
error: module hibernate.core reads package javax.transaction.xa from both jboss.transaction.api.1.2.spec and java.sql
.../src/main/java/module-info.java:1: error: module eu.com.x reads package javax.transaction.xa from both java.sql and jboss.transaction.api.1.2.spec

我可以通过检查我的项目传递依赖项(“gradle dependencies”或“mvn dependency:tree”可能会有所帮助)并通过类似于以下代码的排除来摆脱拆分包编译问题:

configurations.all {
    exclude group: 'org.jboss.spec.javax.transaction', module: 'jboss-transaction-api_1.2_spec'
}

或者

<dependencies>
    <dependency>
      <groupId>org.hibernate</groupId>
      <artifactId>hibernate-core</artifactId>
      <version>5.2.10.Final</version>
      <exclusions>
        <exclusion>
          <groupId>org.jboss.spec.javax.transaction</groupId>
          <artifactId>jboss-transaction-api_1.2_spec</artifactId>
        </exclusion>
      </exclusions> 
    </dependency>
  </dependencies>

我的问题不需要重新包装jar。#JDK8 上没有出现此问题。可能排除依赖项对每个项目都没有帮助。

于 2017-08-13T12:38:10.500 回答
1

从 javaee 和 java.transaction.xa 读取包的 java.transaction.xa 时,我遇到了同样的问题。我通过将此行添加到我的 modul-info.java 来修复它

 opens javax.transaction.xa;

它工作正常,但出现提示说包 javax.transaction.xa 为空或不存在。但是源代码编译正确。

于 2018-10-30T15:41:57.230 回答