jlink是Java 9及更高版本中提供的模块化工具,属于JRE的精简版实现方案,它的核心作用是让开发者可以根据自身应用的实际模块依赖,自定义生成最小化的Java运行环境,避免打包完整的JRE导致资源浪费。传统的完整JRE包含了Java平台的所有模块,而很多应用实际只会用到其中一小部分,jlink正好解决了这个痛点。

jlink的核心原理
Java 9之后引入了模块化系统,将Java平台拆分为多个独立的模块,每个模块都有明确的职责和依赖关系。jlink的工作逻辑就是先分析目标应用依赖的所有模块,再把这些模块以及它们的传递依赖整合起来,生成一个只包含必要内容的运行时镜像,这个镜像可以直接用来运行对应的Java应用,不需要额外安装完整的JRE。
使用jlink的前置条件
- 安装JDK 9及以上版本,因为jlink工具是JDK自带的,位于JDK的bin目录下。
- 目标应用已经模块化,或者能够明确知道应用运行需要依赖哪些Java模块。
- 如果是非模块化的传统应用,需要先通过工具分析其依赖的模块,再进行后续操作。
分析应用的模块依赖
如果是已经模块化的应用,可以直接查看模块描述文件module-info.java中声明的依赖。如果是非模块化的应用,可以使用JDK自带的jdeps工具来分析依赖,示例命令如下:
# 分析target目录下的myapp.jar的依赖,输出模块依赖信息 jdeps --module-path target/libs -s target/myapp.jar
执行命令后会输出应用依赖的所有Java模块,比如可能输出java.base、java.sql等,这些就是后续生成运行时镜像需要包含的基础模块。
使用jlink生成最小运行时镜像
假设我们通过分析得到应用依赖的模块是java.base和java.sql,那么可以使用以下jlink命令生成自定义运行时镜像:
# 生成最小化运行时镜像,输出到custom_jre目录 jlink --module-path $JAVA_HOME/jmods --add-modules java.base,java.sql --output custom_jre --strip-debug --no-header-files --no-man-pages --compress=2
命令参数说明:
--module-path:指定模块路径,这里指向JDK安装目录下的jmods目录,里面存放了所有Java模块的定义文件。--add-modules:指定需要包含的模块,多个模块用逗号分隔。--output:指定生成的运行时镜像的输出目录。--strip-debug:移除调试信息,进一步减小镜像体积。--no-header-files:不包含头文件。--no-man-pages:不包含手册页。--compress=2:开启zip压缩,压缩级别为2,最大程度减小体积。
验证生成的运行时镜像
生成镜像后,可以进入custom_jre/bin目录,使用生成的java命令来运行应用,验证镜像是否可用:
# 使用自定义JRE运行应用 custom_jre/bin/java -jar target/myapp.jar
如果应用能够正常启动运行,说明生成的精简版JRE是符合要求的。此时可以查看custom_jre目录的大小,对比完整JRE的体积,会发现精简后的体积大幅减小。
常见注意事项
- 如果应用依赖第三方库,需要确认第三方库是否有模块化的版本,或者是否能够通过
--add-modules添加对应的依赖,避免运行时出现类找不到的错误。 - 如果应用使用了反射等动态加载类的机制,可能需要额外添加一些模块,或者通过
--add-exports等参数开放模块内部的包访问权限。 - 不同JDK版本的模块可能存在差异,生成镜像时使用的JDK版本最好和运行应用的目标环境JDK版本一致,避免兼容性问题。