JPMS/Jigsaw模块中缺少主类

人气:150 发布:2023-01-03 标签: java maven executable-jar java-9 module-info

问题描述

我正在尝试创建一个模块化的可执行JAR文件,该文件可以在Java 9.0.1上使用java -p <jar file> -m <module>运行。

使用jar cfe test.jar test.Main -C classes/ .创建JAR时,这会正常工作,但在使用mvn packagemvn assembly:single生成JAR时,会抛出module test does not have a MainClass attribute, use -m <module>/<main-class>

这些maven生成的JAR仍然使用java -p test.jar -m test/test.Main工作,所有JAR使用java -jar test.jar工作在类路径上。

我使用jar xf test.jar检查了JAR内容,发现JAR完全相同,只是清单(见下文):

Manifest-Version: 1.0
Created-By: 9.0.1 (Oracle Corporation)
Main-Class: test.Main

Manifest-Version: 1.0
Archiver-Version: Plexus Archiver
Created-By: Apache Maven
Built-By: testuser
Build-Jdk: 9.0.1
Main-Class: test.Main

值得注意的是,在指定工作清单时仍不能使用java -p test.jar -m test

$ jar cfm test.jar test-contents/META-INF/MANIFEST.MF -C classes/ .
$ java -p test.jar -m test

module test does not have a MainClass attribute, use -m <module>/<main-class>

编辑:具有预期行为的回购:https://github.com/deontologic/test

推荐答案

我也不能在jar tool文档中准确地找到这一点,但jmod tool定义得更好一些。

--main-class class-name 

指定要记录在MODULE-info.class文件中的主类。

我可以进一步研究JDK代码,执行不同命令时的行为似乎与ModuleDescriptor.mainClass()mainClass属性有关。

因为在使用--main-class标志打包JAR时,模块描述符中的条目应该用于执行模块,而不需要在执行期间指定主类的完全限定名。

另一方面,这不是真的using the maven jar creation,并且可能会在您发现的未来升级中修复。

有关以下工作原理的更多信息:

java -jar test-1.0.0-SNAPSHOT-jar-with-dependencies.jar
如果指定了-jar选项,则其参数是包含应用程序的类和资源文件的JAR文件的名称。启动类必须由其清单文件(META-INF/MANIFEST.MF)中的Main-Class清单标头指示

指定完全限定名的原因非常明显:

java -p target/test-1.0.0-SNAPSHOT-jar-with-dependencies.jar -m test/test.Main

22