跳到主要内容

手把手将你的Java maven项目通过GraalVM打包成windows可执行程序

1. 背景相关

作为一个程序员,总是会在工作的时候开发一些方便自己工作的程序,但是作为一个Java程序员,给自己或者朋友开发的办公小程序总是使用一个Jar包并不是一个很好的体验,主要是:

  • 启动不方便
  • 需要Java环境
  • 有时候打出来的Jar比较大,不便于分享传输

但是现在有了GraalVM,让这些都不是问题了,直接生成可执行程序,丢哪哪就能用

2. GraalVM

GraalVM是一款高性能的虚拟机,它能够直接将Java程序编译成本地可执行文件,可以在不安装JVM的情况下运行程序。当然了它的特性远不止这些,比如支持多语言,更低的内存占用等等,但这些并不是这篇博客所涉及的,有兴趣可以去他们的官网里了解一下。

安装GraalVM JDK

下载界面:https://github.com/graalvm/graalvm-ce-builds/releases/tag/vm-22.3.1

这里用到的是Java 17版本:

下载完成之后,放入你本地的JDK目录,例如:

接着配置环境变量

简单的通过cmd测试一下

安装native-image

Native Image是一种将Java代码提前编译为独立可执行文件的技术,此刻执行文件包括应用程序类、依赖、运行时库以及JDK静态连接的本机代码。 Graalvm通过子模块SubstrateVM来支持Native Image,相比JVM其生成的程序具有更快的启动时间和更低的运行时开销。

需要在管理员模式下打开一个cmd窗口,然后执行:

gu install native-image

3. 安装Visual studio

为了打包成exe可执行程序,这一步也是不可避免需要做的,这里推荐安装2019版本之后的,我安装的是visual studio 2022,下载地址:https://visualstudio.microsoft.com/zh-hans/vs/

安装完成以后我们还需要一些配置,不然直接进行打包会失败

配置cl.exe到系统的环境变量

安装完Visual studio后,你可以通过我的路径匹配找到你的cl.exe 的路径,并配置cl.exe到系统的环境变量**。**

C:\Program Files (x86)\Microsoft Visual Studio\2022\BuildTools\VC\Tools\MSVC\14.34.31933\bin\Hostx64\x64

如果有问题:可以参考我之前踩过的坑:

Default native-compiler executable 'cl.exe' not found via environment variable PATH

4. 将一个简单的Java应用打包成可执行程序

这里以一个简单的Java应用举例,它的功能是将剪切板上的一大串的xml文本解析为排序后的xml,然后重新放回剪切板,方便我们平时在工作的时候对xml进行对比。

代码已经放在github,有需要的可以自取:https://github.com/MingGH/clip-sort-xml

你可能不信,代码主要部分是ChatGPT写出来的,我只做了获取剪切板内容和放入剪切板这两个方法。。

添加graal-sdk依赖

<dependency>
<groupId>org.graalvm.sdk</groupId>
<artifactId>graal-sdk</artifactId>
<version>21.3.0</version>
<scope>provided</scope>
</dependency>

对maven项目使用跟踪代理生成反射配置

有了代码之后,我们需要对maven项目使用跟踪代理生成反射配置

需要先把你的代码打成jar包,虽然这里并不是SpringBoot项目,但是我使用的是spring-boot-maven-plugin对项目进行打包。

<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<version>3.0.4</version>
</plugin>

可以通过IDEA快速打包,或者通过命令mvn clean package进行打包

构建成功之后,我们进入target目录下:cd target

然后使用跟踪代理运行应用程序:

执行此命令前注意你的idea配置的JDK也需要是GraalVM JDK

java -agentlib:native-image-agent=config-output-dir=META-INF/native-image -jar ./app.jar

执行时,会真实的启动你的项目并且执行,所以最好正常流程和异常流程都要测试到。

执行完成之后,会在target目录下生成一个META-INF的文件夹,这个文件夹中包含的内容就是使用跟踪代理生成的反射配置。

把这个文件夹的内容复制到项目resources目录下

然后回到pom.xml文件,注释掉spring-boot-maven-plugin 添加native-image-maven-plugin ,如下:

<plugin>
<groupId>org.graalvm.nativeimage</groupId>
<artifactId>native-image-maven-plugin</artifactId>
<version>21.2.0</version>
<executions>
<execution>
<goals>
<goal>native-image</goal>
</goals>
<phase>package</phase>
</execution>
</executions>
<configuration>
<skip>false</skip>
<imageName>graalvmMaven</imageName>
<mainClass>run.runnable.clipsortxml.ClipSortXmlApplication</mainClass>
<buildArgs>
--no-fallback
</buildArgs>
</configuration>
</plugin>

通过Native Tools Command Prompt 进行打包

现在接近尾声了,从你的windows搜索窗口,打开Native Tools Command Prompt ,然后进入到你的项目路径。

这个控制台我在执行的时候有点奇怪,只能在C盘执行,所以我把整个项目复制到了C盘的临时目录。

进入对应目录,然后执行:mvn clean package 你就会听到你的电脑起飞了,至少我的电脑是风扇狂转准备起飞了。

当出现build Success的时候就表示打包成可执行程序已经完成。

当中可能会碰到一些错误,我已经替你踩过一次坑了,所以可以直接参考我的解决办法:

native-image - fatal error C1034: stdio.h: 不包括路径集

native-image 抛错fatal error 1083

Default native-compiler executable 'cl.exe' not found via environment variable PATH

完成上面的步骤之后,你会看到在target目录,哇,真的有一个.exe文件,迫不及待去点开试试你的功能。

但是把,如果你是直接使用的我的这个项目,那么快乐还没有来的这么快。

双击之后你会发现一个黑色窗口飘过,然后啥也没发生,其实是因为代码执行出错了

解决Default flavor mapping not found错误

我们先按照正常方式进行排查,任意cmd窗口执行刚刚打包的那个.exe文件,就可以看到具体的错误信息:

这个错误也已经解决:

native-image:Default flavor mapping not found

这里再重复一下:

这是agentlib native-image-agent.exe的bug,已经有人反馈在GraalVM的issues中,你可以在这个链接中发现其他人也碰到了这个问题:https://github.com/oracle/graal/issues/5369

解决办法也很简单,在你通过跟踪代理 生成的META-INF/native-image/resource-config.json中的includes 添加一行:{ "pattern":"\\Qsun/datatransfer/resources/flavormap.properties\\E" }

保存然后重新执行mvn clean package

这时打包出来的.exe文件才是真正可用的,这样一来一个exe文件,更方便丢给不懂代码的朋友了。

5. 参考内容

[native-image] Swing application cannot be compiled on Windows

[native-image] makeShimDLL for java.dll is missing exports and fails

[native-image] java.home property not set

swing JTextField ctrl+v Throw Exception: java.lang.InternalError: Default flavor mapping not found

GraalVM Native Image 快速入门

native-image - fatal error C1034: stdio.h: 不包括路径集

GraalVM与Spring Native初体验,一个让你的应用在100ms内启动的神器

native-image:Default flavor mapping not found

native-image 抛错fatal error 1083

Default native-compiler executable 'cl.exe' not found via environment variable PATH

native-image:Default flavor mapping not found

Could not find agent library native-image-agent on the library path

ChatGPT