首页 文章

用Gradle和Kotlin构建一个可自我执行的jar

提问于
浏览
33

我已经编写了一个简单的kotlin源文件以便开始使用,还有一个gradle脚本文件 . 但我无法弄清楚如何将主要功能添加到清单,以便jar可以自行执行 .

这是我的build.gradle脚本:

buildscript {
    repositories {
        mavenCentral()
    }
    dependencies {
        classpath 'org.jetbrains.kotlin:kotlin-gradle-plugin:0.9.66'
    }
}
apply plugin: "kotlin"
repositories {
    mavenCentral()
}
dependencies {
    compile 'org.jetbrains.kotlin:kotlin-stdlib:0.9.66'
}

jar {
    manifest {
        attributes 'Main-Class': 'com.loloof64.kotlin.exps.ExpsPackage'
    }
}

这是我的com.loloof64.kotlin.exps.Multideclarations.kt

package com.loloof64.kotlin.exps

class Person(val firstName: String, val lastName: String) {
    fun component1(): String {
        return firstName
    }
    fun component2(): String {
        return lastName
    }
}

fun main(args: Array < String > ) {
    val(first, last) = Person("Laurent", "Bernabé")
    println("First name : $first - Last name : $last")
}

当我从终端(java -jar MYJar.jar)启动jar时,我得到以下stacktrace,告诉我kotlin反射库类缺失,实际上它们还没有添加到jar中 . 似乎我错过了最终jar中的kotlin-compiler工件类,以及kotlin-stdlib源代码,但我不知道如何调整gradle构建 .

$> java -jar build/libs/kotlin_exps.jar 

Exception in thread "main" java.lang.NoClassDefFoundError: kotlin/reflect/jvm/internal/InternalPackage
    at com.loloof64.kotlin.exps.ExpsPackage.<clinit>(MultiDeclarations.kt)
Caused by: java.lang.ClassNotFoundException: kotlin.reflect.jvm.internal.InternalPackage
    at java.net.URLClassLoader$1.run(URLClassLoader.java:372)
    at java.net.URLClassLoader$1.run(URLClassLoader.java:361)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.net.URLClassLoader.findClass(URLClassLoader.java:360)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
    at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
    ... 1 more

我正在使用Kotlin 0.9.66和gradle 2.1

4 回答

  • 27

    我找到了解决方法(感谢MkYong website

    • gradle脚本需要将kotlin-compiler工件作为依赖项

    • gradle脚本需要一种方法来收集所有kotlin文件并将它们放入jar中 .

    所以我得到以下gradle脚本:

    buildscript {
        repositories {
            mavenCentral()
        }
        dependencies {
           classpath 'org.jetbrains.kotlin:kotlin-gradle-plugin:1.0.1-2'
        }
    }
    
    apply plugin: "kotlin"
    
    repositories {
        mavenCentral()
    }
    
    dependencies {
        compile 'org.jetbrains.kotlin:kotlin-stdlib:1.0.1-2'
    }
    
    jar {
        manifest {
            attributes 'Main-Class': 'com.loloof64.kotlin.exps.MultideclarationsKT'
        }
    
        // NEW LINE HERE !!!
        from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
    }
    
  • 1

    如果你碰巧像我一样愚蠢:

    不要将IntelliJ运行配置创建为应用程序 . IntelliJ将假设它是Java并且它永远不会工作 . 而是使用“Kotlin”条目 .

  • 22

    添加插件 application ,然后将 mainClassName 设置为

    mainClassName = '[your_namespace].[your_arctifact]Kt'
    

    例如,假设您已将以下代码放在名为 main.kt 的文件中:

    package net.mydomain.kotlinlearn
    
    import kotlin
    import java.util.ArrayList
    
    fun main(args: Array<String>) {
    
        println("Hello!")
    
    }
    

    你的 build.gradle 应该是:

    apply plugin: 'kotlin'
    apply plugin: 'application'
    
    mainClassName = "net.mydomain.kotlinlearn.MainKt"
    

    实际上,Kotlin正在构建一个类来封装您的主要函数,该函数使用与文件相同的名称命名 - 使用Title Case .

  • 0

    谢谢,对我而言,它添加了jar部分

    jar {
        manifest {
            attributes 'Main-Class': 'com.photofiles.Application'
        }
    
        from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
    }
    

    不需要应用程序插件 .

相关问题