我正在用 Gradle 构建一个 JAR 文件。当我尝试运行它时,出现以下错误
没有主要清单属性,在 RxJavaDemo.jar
我尝试操作 manifest
属性,但我想我忘记添加依赖项或其他内容。我到底做错了什么?
apply plugin: 'java'
apply plugin: 'application'
mainClassName = 'demo.MainDashboard'
dependencies {
compile files ("H:/Processes/Development/libraries/hikari-cp/HikariCP-2.4.1.jar")
compile files ("H:/Processes/Development/libraries/controls-fx/controlsfx.jar")
compile files ("H:/Processes/Development/libraries/database_connections/sqlite-jdbc-3.8.6.jar")
compile files ("H:/Processes/Development/libraries/guava/guava-18.0.jar")
compile files ("H:/Processes/Development/libraries/rxjava/rxjava-1.0.12.jar")
compile files ("H:/Processes/Development/libraries/rxjava-extras/rxjava-extras-0.5.15.jar")
compile files ("H:/Processes/Development/libraries/rxjavafx/RxJavaFX-1.0.0-RC1-SNAPSHOT.jar")
compile files ("H:/Processes/Development/libraries/rxjavaguava/rxjava-guava-1.0.3.jar")
compile files ("H:/Processes/Development/libraries/rxjava-jdbc/rxjava-jdbc-0.6.3.jar")
compile files ("H:/Processes/Development/libraries/slf4j/slf4j-api-1.7.12.jar")
compile files ("H:/Processes/Development/libraries/tom-commons/tom-commons.jar")
}
sourceSets {
main.java.srcDir "src/main/java"
main.resources.srcDir "src/main/resources"
}
jar {
manifest {
attributes(
"Class-Path": configurations.compile.collect { it.getName() }.join(' '))
}
from configurations.compile.collect { entry -> zipTree(entry) }
}
尝试更改清单属性,例如:
jar {
manifest {
attributes(
'Class-Path': configurations.compile.collect { it.getName() }.join(' '),
'Main-Class': 'hello.HelloWorld'
)
}
}
然后只需将 'hello.helloWorld'
更改为 '<your packagename>.<the name of your Main class>'
(您的 Main 类有一个 main 方法)。在这种情况下,您在清单中创建一个指向此类的属性,然后运行一个 jar。
要使 jar
文件可执行(以便 java -jar
命令起作用),请在 MANIFEST.MF
中指定 Main-Class
属性。
在 Gradle 中,您可以通过配置 jar
任务来完成。
对于 Groovy DSL,请参阅这些答案 ([1], [2])
对于 Kotlin DSL,您可以使用以下代码片段:
tasks.withType<Jar> {
manifest {
attributes["Main-Class"] = "com.caco3.Main"
}
}
为什么 mainClassName 不能按预期工作?
或者为什么 mainClassName
没有在清单中指定属性?
mainClassName
属性来自 application
plugin。插件:
使在开发过程中本地启动应用程序变得容易,并将应用程序打包为 TAR 和/或 ZIP,包括操作系统特定的启动脚本。
所以 application
插件的目的不是生成可执行的 jar
s
当设置 mainClassName
属性时,则:
./gradlew run 将在属性中指定的类中启动 main 方法。使用 distZip/distTar 任务构建的 zip/tar 存档将包含一个脚本,该脚本将启动指定的先前类的 main 方法。
这是设置主类的shell脚本行:
$ grep Main2 gradletest
eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLETEST_OPTS -classpath "\"$CLASSPATH\"" com.caco3.gradletest.Main2 "$APP_ARGS"
FWIW - 我使用以下 jar 任务将所有编译依赖项组装到 jar 文件中,并使用上述建议来正确设置类路径
apply plugin: 'java-library'
jar {
manifest {
attributes(
'Class-Path': configurations.compile.collect { it.getName() }.join(' '),
'Main-Class': 'your.main.class.goes.here'
)
}
// You can reference any part of the dependency configurations,
// and you can have as many from statements as you need
from configurations.compile
// I just copied them into the top of the jar, so it looks like the eclipse exported
// runnable jar, but you could designate a lib directory, and reference that in the
// classpath as "lib/$it.name" instead of it.getName()
into ''
}
作为对 Denis Zavedeev answer 的补充,以下是 Kotlin DSL (build.gradle.kts) 的更多方法:
tasks.jar {
manifest.attributes["Main-Class"] = "com.example.MyMainClass"
}
tasks.jar {
manifest {
attributes["Main-Class"] = "com.example.MyMainClass"
}
}
旁注:要创建可运行的胖 JAR(也称为 uber JAR),请参阅 this post。
collect {}
部分才能让它为我工作。您的代码假定所有依赖项都与您的主类位于同一文件夹中。compile
。改为使用:'Class-Path': configurations.runtimeClasspath.files.collect { it.getName() }.join(' ')