问题描述
我正在尝试创建一个模块化的可执行JAR文件,该文件可以在Java 9.0.1上使用java -p <jar file> -m <module>运行。
使用jar cfe test.jar test.Main -C classes/ .创建JAR时,这会正常工作,但在使用mvn package和mvn 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