以 Native 的方式运行 Java 程序

1519

介绍

Java 由于其机制, 其 Java 程序通常都是由 JVM 来解释运行, 所以运行 Java 程序必须使用 java -jar 或其他命令来运行, 而不能在没有 JVM 的情况下运行

通过 GraalVM 可以使用 native-image 命令将 Java 的 class, module, jar 编译成一个二进制 native 程序, 从而可以不需要 JVM 直接运行

操作方法

通常我们都希望将一个有用的程序进行 native 编译, 所以这里不会讲如何编译 "hello world" 等代码

官方文档: https://www.graalvm.org/latest/guides/

下载 GraalVM

社区版 GraalVM JDK: https://www.oracle.com/java/technologies/downloads/#graalvmjava21

和正常的 JDK 一样的使用方式

使用工具生成配置

由于 Java 的动态代理, 反射是在运行中确定的, native-image 是基于静态分析的, 所以必须提前告知其反射等信息

使用 java -agentlib 来生成配置文件

例如在当前目录下, 生成 demon.jar 的配置文件:

java -agentlib:native-image-agent=config-output-dir=./config -jar demo.jar 

实际上他是把你的程序运行一遍分析

生成后的文件如下:

config
├── jni-config.json
├── predefined-classes-config.json
├── proxy-config.json
├── reflect-config.json
├── resource-config.json
└── serialization-config.json

将这些文件全部复制到项目的 resources/META-INF/org.example/demo 下 (注意换成你自己的包名)

复制后的文件结构如下:

native-demo                                                                     
├── build.gradle.kts
└── src
    ├── main
        ├── java
        │   └── org
        │       └── example
        │           └── demo
        │               ├── Main.java
        └── resources
            └── META-INF
                └── native-image
                    └── org.example
                        └── demo
                            ├── jni-config.json
                            ├── predefined-classes-config.json
                            ├── proxy-config.json
                            ├── reflect-config.json
                            ├── resource-config.json
                            └── serialization-config.json

命令行编译

在完成上一步后重新打包 jar

使用 native-image -jar 编译 jar

native-image -jar demo.jar   

编译完成就可以得到一个可运行的二进制文件了

构建工具编译

gradle

官方文档: https://graalvm.github.io/native-build-tools/latest/gradle-plugin.html

此处使用 KotlinDSL

首先添加插件

plugins {
    id("org.graalvm.buildtools.native") version "0.10.1"
}

添加配置

graalvmNative {
    binaries {
        named("main") {
            imageName.set("demo")
            mainClass.set("org.example.demo.Main")
            useFatJar.set(true)
        }
    }
    binaries.all {
        buildArgs.add("--verbose")
    }
}

添加完成之后就可以使用 gradle nativeCompile 编译, 并且可以使用 gradle nativeRun 直接运行

maven

与 gradle 类似

官方文档: https://graalvm.github.io/native-build-tools/latest/maven-plugin.html

注意事项

  • 编译运行后提示 "ClassNotFound" 一般是反射配置问题
  • 编译运行后提示 "UnsupportedFeature" 可能是你使用了某些不兼容的库, 比如 log4j2
  • 社区版为 Serial GC, 需要细心配置 vm 选项