kotlin编译失败_Kotlin使用GraalVM开发原生命令行应用
背景
之前用kotlin開發過一款根據建表DDL語句生成plantuml ER圖的應用。被問如何使用,答曰"給你一個jar包,然后執行java -jar ddl2plantuml.jar ./ddl.sql ./er.puml 就可以了。是不是so easy?"
結果被吐槽了一番,
這個吐槽帶來了一個思考: 為什么Java很少用于開發原生命令行(CLI)應用呢?我認為主要問題有2個
方案
為了解決上述問題,引入2個名詞
Picocli
Picocli 致力于提供“最簡便的方式來創建富命令行應用,這種應用可以在 JVM 上和 JVM 之外運行”
使用起來非常簡單
fun main(args: Array) { val cmd = CommandLine(Convert()) when { args.isEmpty() -> { cmd.usage(System.out) } else -> { val exitCode = cmd.execute(*args) exitProcess(exitCode) } }}@CommandLine.Command(name = "ddl2plantuml", version = ["軟件名稱:Ddl2plantuml版本:V1.1.0"], description = ["convert sql ddl to plantuml er"], mixinStandardHelpOptions = true)class Convert : Callable { @CommandLine.Parameters(index = "0", description = ["The sql ddl file that should be convert to plantuml er."]) lateinit var src: Path @CommandLine.Option(names = ["-o", "--output"], description = ["The file where the plantuml file to be saved. default is console "]) private var target: Path? = null override fun call(): Int { require(src.toFile().exists()) { "ddl file must be existed!" } when (target) { null -> { FileReader(src).read() .apply { ConsoleWriter(this).write() } } else -> { FileReader(src).read() .apply { FileWriter(target!!, this).write() } } } return 0 }}效果
這里介紹用到的幾個注解及概念
- @Parameters 和 @Options 都是用來定義參數,區別在于 @Parameters根據位置區分,而@Options可以指定名稱
- 退出碼。call()方法返回的0表示退出碼,用來描述命令行應用的執行結果。通常用0表示成功,其他數字為自定義異常。退出碼不會影響程序的執行,但是有一個很實用的功能是當你通過連接的方式同時執行多個應用時,一個非零的退出碼會中斷這個組合。如: ./ddl2plantuml_mac ddl.sql |grep "table"
- 版本及幫助信息。可以自定義并指定樣式,version可以通過versionProvider自定義生成。
GraalVM
Go的一個宣傳點是可以將程序編譯為一個靜態可執行文件,而Java也可以通過GraalVM做到這一點
GraalVM: Run Programs Faster Anywhere
這個slogan和Java的"Write Once, Run Anywhere"遙相呼應,同時又展示了極大的野心,準備帶來下一個20年的輝煌。
GraalVM 是一個高性能的通用虛擬機,可以運行使用 JavaScript,Python 3,Ruby,R,基于 JVM 的語言以及基于 LLVM 的語言開發的應用。 GraalVM 消除了編程語言之間的隔離性,并且通過共享運行時增強了他們的互操作性。它可以獨立運行,也可以運行在 OpenJDK,Node.js,Oracle,MySQL 等環境中。
可以看到GraalVM提供了非常強大的功能,這里我們不做展開介紹,只看如何解決我們遇到的問題。主要用到了2個功能特性
使用方式
編譯后的native image不運行在Java VM上,但是包含了必要的組件,如內存管理和線程調度,這些組件來自另一個Substrate VM。這個過程稱為提前編譯
此時我們已經得到了一個可以直接執行的原生命令行應用
./ddl2plantuml_mac ddl.sql注意:
native image不支持Java的所有特性,尤其是對reflection的限制。在這次改造過程中,原來通過阿里的druid進行sql解析,但是druid使用了大量的reflection導致native image編譯失敗,所以改用jsqlparser。
其他
總結
以上是生活随笔為你收集整理的kotlin编译失败_Kotlin使用GraalVM开发原生命令行应用的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Web前端-html页面-网易注册表单,
- 下一篇: 第二次作业:软件分析之网易云音乐