我最近从旧的 1.2 多平台切换到 1.3。不同之处在于,每个多平台模块都有一个 build.gradle 文件(我有 5 个),因此配置少了很多。 但是,我似乎无法配置使用 jvm 平台的所有依赖项创建可运行的 fat jar。 我曾经在我的 jvm 项目和 jar 任务中使用标准的“应用程序”插件,但这不再起作用了。我发现有“jvmJar”任务,我修改了它(设置主类),但创建的 jar 不包含依赖项,并且在 ClassNotFoundException 上崩溃。我该怎么做?
这就是我现在拥有的:
jvm() {
jvmJar {
manifest {
attributes 'Main-Class': 'eu.xx.Runner'
}
from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
}
}
我确实遇到了那个障碍并使用了这个解决方法。
让我们称您的项目为
Project
。
创建另一个子模块,例如
subA
,它将具有 gradle 符号 Project:subA
现在,
subA
的 build.gradle 中有你的多平台代码(它是带有 apply :kotlin-multiplafrom 的 gradle 项目)
创建另一个仅针对 jvm 的子模块,例如
subB
,它将具有 gradle 符号 Project:subB
所以,
subB
将有插件:'application'
和'org.jetbrains.kotlin.jvm'
plugins {
id 'org.jetbrains.kotlin.jvm' version '1.3.31'
id "application"
}
apply plugin: "kotlinx-serialization"
group 'tz.or.self'
version '0.0.0'
mainClassName = "com.example.MainKt"
sourceCompatibility = 1.8
compileKotlin {
kotlinOptions.jvmTarget = "1.8"
}
dependencies {
implementation project(':subA')
}
您可以像常规 java 项目一样继续构建 subB,甚至使用现有的插件,它会起作用
让 gradle/多平台工作的唯一方法似乎是无休止的尝试和错误;这是一场噩梦,与其说它是一个“构建”系统,不如说它是一个“构建系统”;换句话说,这两个工具(一起或单独)只是实现插件制造商想要的单个软件开发生命周期的一种手段,但是,如果您已经设计了所需的软件生命周期和 CI/CD 系统,并且现在您尝试实现该工程,使用这些工具比使用脚本、代码或 Maven 来实现要困难得多。造成这种情况的原因有很多:
因此,在撰写本文时,这似乎是使用 kotlin 1.3.72、multiplatform 1.3.72、ktor 1.3.2 和 gradle 6.2.2(使用 kts 格式)时使用的正确语法。
注意 fatJar 似乎可以正确组装,但无法运行,它找不到该类,因此我添加了我同时使用的第二个 runLocally 任务。
这不是一个完整的解决方案,所以我讨厌将其发布在这里,但据我所知......这是我可以在任何地方找到记录的最完整和最新的解决方案。
//Import variables from gradle.properties
val environment: String by project
val kotlinVersion: String by project
val ktorVersion: String by project
val kotlinExposedVersion: String by project
val mySqlConnectorVersion: String by project
val logbackVersion: String by project
val romeToolsVersion: String by project
val klaxonVersion: String by project
val kotlinLoggingVersion: String by project
val skrapeItVersion: String by project
val jsoupVersion: String by project
val devWebApiServer: String by project
val devWebApiServerVersion: String by project
//Build File Configuration
plugins {
java
kotlin("multiplatform") version "1.3.72"
}
group = "com.app"
version = "1.0-SNAPSHOT"
repositories {
mavenCentral()
jcenter()
jcenter {
url = uri("https://kotlin.bintray.com/kotlin-js-wrappers")
}
maven {
url = uri("https://jitpack.io")
}
}
//Multiplatform Configuration
kotlin {
jvm {
compilations {
val main = getByName("main")
tasks {
register<Jar>("buildFatJar") {
group = "application"
manifest {
attributes["Implementation-Title"] = "Gradle Jar File Example"
attributes["Implementation-Version"] = archiveVersion
attributes["Main-Class"] = "com.app.BackendAppKt"
}
archiveBaseName.set("${project.name}-fat")
from(main.output.classesDirs, main.compileDependencyFiles)
with(jar.get() as CopySpec)
}
register<JavaExec>("runLocally") {
group = "application"
setMain("com.app.BackendAppKt")
classpath = main.output.classesDirs
classpath += main.compileDependencyFiles
}
}
}
}
js {
browser { EXCLUDED FOR LENGTH }
}
sourceSets { EXCLUDED FOR LENGTH }
}
让它与 kotlin 1.3.61 中的多平台插件一起使用:
以下内容适用于
src/jvmMain/kotlin/com/example/Hello.kt
中的主文件
Hello.kt 还必须将其包指定为
package com.example
我以这种方式配置了我的 jvm 目标:
kotlin {
targets {
jvm()
configure([jvm]) {
withJava()
jvmJar {
manifest {
attributes 'Main-Class': 'com.example.HelloKt'
}
from { configurations.runtimeClasspath.collect { it.isDirectory() ? it : zipTree(it) } }
}
}
}
}
让它与 luca992 的稍微修改版本一起工作:
kotlin {
jvm() {
withJava()
jvmJar {
manifest {
attributes 'Main-Class': 'sample.MainKt'
}
from { configurations.runtimeClasspath.collect { it.isDirectory() ? it : zipTree(it) } }
}
}
...
}
建议的答案对我来说不起作用,可能是 kotlin 插件 API 发生了变化。我使用 kotlin 2.0 插件进行以下设置:
build.gradle.kts
:
import com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar
import org.jetbrains.kotlin.gradle.ExperimentalKotlinGradlePluginApi
plugins {
alias(libs.plugins.kotlinMultiplatform)
alias(libs.plugins.shadow)
}
group = "com.funbiscuit"
version = "1.0-SNAPSHOT"
repositories {
mavenCentral()
}
kotlin {
val javaMainClass = "MainKt"
jvm {
@OptIn(ExperimentalKotlinGradlePluginApi::class)
mainRun {
mainClass = javaMainClass
}
withJava()
tasks.register<ShadowJar>("jvmShadowJar") { // create fat jar task
val mainCompilation = compilations["main"]
val jvmRuntimeConfiguration = mainCompilation
.runtimeDependencyConfigurationName
.let { project.configurations[it] }
from(mainCompilation.output.allOutputs) // allOutputs == classes + resources
configurations = listOf(jvmRuntimeConfiguration)
archiveClassifier.set("fatjar")
manifest.attributes("Main-Class" to javaMainClass)
}
}
sourceSets {
commonMain.dependencies {
}
jvmMain.dependencies {
}
}
}
gradle/libs.versions.toml
:
[versions]
kotlin = "2.0.0"
shadow = "8.3.4"
[plugins]
kotlinMultiplatform = { id = "org.jetbrains.kotlin.multiplatform", version.ref = "kotlin" }
shadow = { id = "com.gradleup.shadow", version.ref = "shadow" }
src/commonMain/kotlin/Main.kt
:
fun main() {
println("Hello!")
}