缘起
云计算时代的来临,有一门被誉为云计算伴生编程语言:golang
其与docker的天生的搭配,开启了云计算的黄金时代。
然而java确实一门不服老的语言,不断的进化着,springboot 4 和spring 7的发布,尤其是springboot4 ,GraalVM 25 直接从实验特性转向了生产。虚拟线程常态化也是值得学习的特性。
加上还有很多的老的项目还在用springboot技术,所以想把java的整个技术栈再捡起来学习一遍:后续会陆续写一些关于springboot4 、jkd 25 lts的一些文。
学到哪儿就写到哪儿:今天想试试springboot程序打包docker的几种方式的大小区别,探索一下最佳的方式!
开始
准备一个最简单的restful的程序,使用spring initializr 生成项目,基础配置如下:

undefined
注意2点:1、java 25 2、加入graalvm
一个hello world
kotlin
package com.example.demo.controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
class HelloController {
@GetMapping("/hello")
String sayHello(){
return "hello world!";
}
}
显示编译一下代码,看看原始jar包有多大:
bash
./gradlew build
一个包含mvc基本程序的一个hello world,大小79M

undefined
方式一:全默认方式的打包镜像
不做任何更多的配置,只使用springboot 4 和gradle的默认配置:
根据资料,springboot的默认使用paketobuildpacks/builder-jammy-tiny 镜像,这是一个以ubuntu 22为基础的镜像,大小50-70M:
注意:犹豫我们生成项目配置的时候加入了graalm支持,必须先注释调,否则默认会使用graal技术:
build.gradle 文件部分:
bash
plugins {
id 'java'
id 'org.springframework.boot' version '4.0.3'
id 'io.spring.dependency-management' version '1.1.7'
//id 'org.graalvm.buildtools.native' version '0.11.4'
使用命令打包:
ini
./gradlew bootBuildImage --imageName=hello:normal
大小337M,同时也可以看到打包下载的基础镜像:

undefined
方式二: 默认配置graalvm打包
加入graalvm的配置(方式一注释调的部分),打包:
ini
./gradlew bootBuildImage --imageName=hello:gralvm
大小123M:

undefined
方式三: 手工打包,使用标准openjdk镜像为基础
所谓手工打包,就是自己写dockerfile,自己设置基础镜像,先试试默认情况下常用的openjdk镜像
bash
FROM openjdk:25-jdk-slim
WORKDIR /app
RUN useradd -U spring
USER spring:spring
COPY build/libs/*.jar app.jar
EXPOSE 8080
CMD ["java", "-jar", "app.jar"]
打包命令:
bash
docker build -t hello:openjdk -f deploy/openjdk.Dockerfile .
大小:505M

undefined
方式四: 手工打包,使用jlink构建一个最小的运行jre环境,以alpine为基础
虽然现在的java不再单独提供jre环境了,但是可以通过jlink构建一个程序所需的最小运行环境,然后部署到alpine linux上
bash
FROM eclipse-temurin:25-jdk-alpine AS jre-builder
RUN apk update && apk add binutils
RUN $JAVA_HOME/bin/jlink \
--verbose \
--add-modules java.base \
--strip-debug \
--no-man-pages \
--no-header-files \
--compress=2 \
--output /optimized-jdk-25
FROM alpine:latest
ENV JAVA_HOME=/opt/jdk/jdk-25
ENV PATH="${JAVA_HOME}/bin:${PATH}"
COPY --from=jre-builder /optimized-jdk-25 $JAVA_HOME
RUN addgroup --system spring && adduser --system spring --ingroup spring
RUN mkdir /app && chown -R $APPLICATION_USER /app
COPY --chown=spring:spring build/libs/*.jar /app/app.jar
WORKDIR /app
USER spring
EXPOSE 8080
ENTRYPOINT [ "java", "-jar", "/app/app.jar" ]
构建命令:
bash
docker build -t hello:alpine -f deploy/jlink.Dockerfile .
大小: 93M

undefined
有点不放心,跑一下试试:
arduino
docker run -it -p 8080:8080 hello:alpine

undefined
访问一下:

undefined
一切正常!
小结
以上只是比较了打包大小的不同,这对graalvm是不公平的,毕竟他的特点是增强 AOT(Ahead-of-Time)编译,降低内存占用,提高启动时间,还是得测试一下:
arduino
docker run -it -p 8080:8080 --name alpine hello:alpine
docker run -it -p 9090:8080 --name graalvm hello:graalvm
docker stats
看到资源占用:

undefined
graalvm的内存占用不到原始java代码的1/4!
所以:
1、对应新的项目,高版本jdk项目,尤其是最新的25版本,完全可以考虑以graalvm来编译打包
2、对应旧的项目,或者低版本jdk的项目,生产环境使用alpine定制jre的方式,就是方式4,测试环境就使用默认openjdk:毕竟包含了大量的工具,方便调测。