言成言成啊 | Kit Chen's Blog

双平台GraalVM编译二进制程序

发布于2023-05-17 02:35:12,更新于2024-06-02 14:32:04,标签:java graalvm  转载随意,文章会持续修订,请注明来源地址:https://meethigher.top/blog

本文示例均采用Java11,GraalVM目前无法支持跨平台编译,比如,我通过Linux直接编译Windows可执行的exe,是不行的。

Go语言是可以的

因此,需要掌握两种平台的GraalVM的安装、使用。

一、背景

1.1 为何GraalVM快?

常规Java编译的Jar,为了考虑平台的兼容性,在启动后,将字节码解释为机器码,这个过程叫做JIT编译。所以启动上会慢。

GraalVM编译的二进制程序,在编译时直接将字节码转为了机器码,这个过程叫做AOT编译。所以启动上会快,但是对反射支持的就不友好。

因此Java严格来说既是编译型,又是解释型。只不过编译成的是字节码。

截止到2023年05月,总结GraalVM的弱项

  1. 容量占用:使用Graalvm编译后的exe,首先需要依赖C++的开发环境,并且里面还带了JDK编译后的机器码,所以内存约至少10M以上。
  2. 失去原有特性:如Java跨平台的特性、反射、运行时的动态代理等等。
  3. 编译对Java语法有要求,死板且不够灵活,比如上述第二点中的问题。

综上所述,GraalVM只是为了让Java能做而做,而不是为了Java需要做而做。如果以上问题存在,相信以后也不会有太多开发者使用GraalVM开发。不如直接转C/C++/Go,而Go属于最简单的替代品。

1.2 JIT与AOT对比

JIT(Just-in-Time)编译和AOT(Ahead-of-Time)编译是两种不同的编译策略,它们在代码编译和执行的时间点上有所区别。

  1. JIT(Just-in-Time)编译
    • JIT 编译是在程序运行时动态地将字节码或中间代码(如Java字节码)转换为本机机器码。
    • JIT 编译器根据程序的运行时行为和环境进行优化,并将热点代码(频繁执行的代码)即时编译为本机机器码。
    • JIT 编译器可以根据特定硬件平台的特性进行优化,以提高执行速度和性能。
    • JIT 编译通常用于解释性语言或虚拟机中,例如Java虚拟机(JVM)。
  2. AOT(Ahead-of-Time)编译
    • AOT 编译是在程序运行之前将源代码或中间代码(如字节码)静态编译为本机机器码。
    • AOT 编译器在编译阶段进行优化,根据目标平台的特性生成高效的本机代码。
    • AOT 编译可以提供更快的启动时间和更稳定的性能,因为代码已经被编译为本机机器码,无需再进行即时编译。
    • AOT 编译通常用于静态语言或需要更高性能的应用程序,例如C/C++编译器。

主要区别:

  • JIT 编译是在运行时动态地将代码编译为机器码,而 AOT 编译是在程序运行之前将代码静态编译为机器码。
  • JIT 编译可以根据运行时行为进行优化,而 AOT 编译在编译阶段进行优化。
  • JIT 编译需要在程序运行时进行编译,可能导致启动延迟和即时编译的开销,而 AOT 编译可以提供更快的启动时间和更稳定的性能。

需要注意的是,JIT 编译和 AOT 编译并不是互斥的概念,而是两种不同的编译策略。在某些情况下,可以结合使用它们,例如GraalVM提供了同时支持 JIT 编译和 AOT 编译的能力,使得可以根据应用程序的需求选择最适合的编译方式。

二、Linux

2.1 前置环境

环境:Centos7.9、科学上网

1
yum -y install gcc zlib*

2.2 安装GraalVM-ce-java11

使用 uname 命令:

1
uname -m

该命令将返回当前系统的机器架构信息。如果返回值是 “x86_64”,则表示系统是 AMD64 架构;如果返回值是 “aarch64”,则表示系统是 AArch64 架构。

1
curl -L -o graalvm-ce-java11-linux-amd64-21.3.3.1.tar.gz https://github.com/graalvm/graalvm-ce-builds/releases/download/vm-21.3.3.1/graalvm-ce-java11-linux-amd64-21.3.3.1.tar.gz && tar -zxvf graalvm-ce-java11-linux-amd64-21.3.3.1.tar.gz && cd graalvm-ce-java11-21.3.3.1/bin && ./gu install native-image && ./gu list

执行如图

三、Windows

3.1 前置环境

windows需要预置的C++环境,这边可以自己下,也可以直接使用Visual Studio下载。我的电脑是win11,而且刚好最近编写dll,因此直接使用Visual Studio2022来安装环境。

打开Visual Studio2022-工具-获取工具和功能

之后,需要将cl.exe的所在目录,配置到环境变量PATH中。

3.2 安装GraalVM-ce-java11

详细配置方式可以参照如下内容,两者任选其一即可

执行命令,进行graalvm-jdk11下载

1
curl -L -o graalvm-ce-java11-windows-amd64-21.3.3.1.zip https://github.com/graalvm/graalvm-ce-builds/releases/download/vm-21.3.3.1/graalvm-ce-java11-windows-amd64-21.3.3.1.zip

然后将jdk进行解压,并配置成环境变量,配置成功后验证结果如图所示。

安装native-image

1
gu install native-image

使用Visual Studio自带的编译工具,按照第四步进行编译即可。

直接使用cmd编译会报错native-image - fatal error C1034: stdio.h: 不包括路径集

四、编译二进制程序

源码地址

4.1 编译字节码

创建Test.java

1
2
3
4
5
6
7
import java.text.SimpleDateFormat;
import java.util.Date;
class Test {
public static void main(String[] args) {
System.out.println("hello world! now time is "+new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()));
}
}

编译Test.java

1
2
3
4
5
./javac -encoding utf-8 Test.java
# 执行下面这行代码,会发现输出hello world! now time is 2023-05-16 21:10:09
./java Test
# 编译为二进制程序
./native-image Test

如图即为编译成功

移到没有Java环境的路径,执行二进制程序

1
mv test /root/test && cd && chmod +x test && ./test

4.2 编译jar包

4.2.1 编译原生jar包

执行命令

1
2
mvn clean package
native-image -jar pure-demo.jar

在native-image中,即使主线程结束,其他线程一样执行,程序不会退出

4.2.2 编译springboot jar包

保持原来的代码不变基础上,需要引入spring-native和graalvm打包插件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
<dependencies>
<dependency>
<groupId>org.springframework.experimental</groupId>
<artifactId>spring-native</artifactId>
<version>0.9.2</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
</dependencies>

<build>
<plugins>
<plugin>
<groupId>org.springframework.experimental</groupId>
<artifactId>spring-aot-maven-plugin</artifactId>
<version>0.9.2</version>
<executions>
<execution>
<id>test-generate</id>
<goals>
<goal>test-generate</goal>
</goals>
</execution>
<execution>
<id>generate</id>
<goals>
<goal>generate</goal>
</goals>
</execution>
</executions>
</plugin>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<classifier>exec</classifier>
</configuration>
</plugin>
</plugins>
</build>

<profiles>
<profile>
<id>native-image</id>
<build>
<plugins>
<plugin>
<groupId>org.graalvm.nativeimage</groupId>
<artifactId>native-image-maven-plugin</artifactId>
<version>21.0.0.2</version>
<configuration>
<!-- native-image需要执行的应用程序的入口点 -->
<mainClass>${start.class}</mainClass>
<buildArgs>
<buildArg>--enable-https</buildArg>
</buildArgs>
</configuration>
<executions>
<execution>
<goals>
<goal>native-image</goal>
</goals>
<phase>package</phase>
</execution>
</executions>
</plugin>
</plugins>
</build>
</profile>
</profiles>

执行命令,直接打包成镜像

1
mvn clean package -Pnative-image

放一张原生jar的运行耗时。

GraalVM牛逼!

五、参考致谢

  1. GraalVM Native Image: Hello World - YouTube
  2. 官方文档
  3. Spring Tips: The GraalVM Native Image Builder Feature
  4. 参考第3条的视频:将SpringBoot2.3+jdk8应用编译为二进制
  5. 使用GraalVM将Java编译成本机可执行程序_哔哩哔哩_bilibili
  6. 手把手将你的Java maven项目通过GraalVM打包成windows可执行程序 - Asher的博客
发布:2023-05-17 02:35:12
修改:2024-06-02 14:32:04
链接:https://meethigher.top/blog/2023/graalvm-native-image/
标签:java graalvm 
付款码 打赏 分享
shift+ctrl+1可控制目录显示