时髦的 Java 9 模块支持

我花了一些时间将用Groovy编写的项目迁移到Java 10。现在可以编译并运行它了。但它仍然没有使用Java 9模块化的任何好处。
谷歌搜索Groovy和Java 9模块几乎没有什么。

那么,是否可以将Groovy项目迁移到JDK 10和Project Jigsaw模块一起使用呢?


答案 1

好吧,经过几天的实验,我想出了答案 - 是的,可以将Groovy与Project Jigsaw模块一起使用。
但这需要一些额外的努力。

假设我们有以下文件结构:

├── build
├── jigsaw
│   └── module
│       └── test
│           └── Application.groovy
├── lib
│   └── groovy.all.jar
└── module-info.java  

模块信息.java

module main {
    requires groovy.all;
}

Application.groovy

package jigsaw.module.test

class Application {
    static void main(String[] args) {
        println "Hello module!"
    }
}

首先,我们需要使用javac编译module-info.java文件,而不是使用groovyc编译所有文件,因为groovy将模块文件视为闭包。

让我们来做吧:

javac -d build --module-path lib/ module-info.java

--module-path 将包含我们的 groovy.all.jar作为自动模块,其名称派生自 JAR 文件名。

接下来我们需要编译 Application.groovy

groovyc -d build jigsaw/module/test/Application.groovy

一切进展顺利。
编译后,我们有模块信息.class(又名模块描述符)和应用程序.class

├── build
│   ├── jigsaw
│   │   └── module
│   │       └── test
│   │           └── Application.class
│   └── module-info.class
├── jigsaw
│   └── module
│       └── test
│           └── Application.groovy
├── lib
│   └── groovy.all.jar
└── module-info.java

现在,让我们尝试运行已编译的模块。

java --module-path build:lib --module main/jigsaw.module.test.Application

这就是我们得到的

Error occurred during initialization of boot layer
java.lang.module.FindException: Unable to derive module descriptor for lib/groovy.all.jar
Caused by: java.lang.module.InvalidModuleDescriptorException: Provider class moduleName=groovy-all not in module

这是什么意思?我不知道。经过大量的谷歌搜索,我发现了类似的东西

因此,我们需要从JAR中手动删除这些文件:

  • /META-INF/services/org.codehaus.groovy.source.Extensions
  • /META-INF/services/org.codehaus.groovy.runtime.ExtensionModule

最后,我们的 Java 模块能够启动

java --module-path build:lib --module main/jigsaw.module.test.Application
Hello module!

所有操作都是使用Oracle JDK 10和Groovy 2.4.15完成的。


答案 2

推荐