探索springboot程序打包docker的最佳方式

缘起

云计算时代的来临,有一门被誉为云计算伴生编程语言: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:毕竟包含了大量的工具,方便调测。

相关推荐
葫芦和十三3 小时前
图解 MongoDB 07|索引类型:七种索引,七种访问形状
后端·mongodb·agent
朦胧之5 小时前
AI 编程-老项目改造篇
java·前端·后端
爱勇宝7 小时前
我做了一个只用来搜歌词的小 App
android·前端·后端
IT_陈寒8 小时前
SpringBoot自动配置坑了我一晚上,原来问题出在这
前端·人工智能·后端
SelectDB9 小时前
Litefuse 开源并推出单进程轻量模式,25 秒就能跑起来的 Agent 可观测与评估平台
运维·后端·自动化运维
SelectDB9 小时前
秒级弹性、最高降本 70%:SelectDB Serverless 如何重塑云数仓资源效率
大数据·后端·云原生
PinkSun9 小时前
Spring AI ChatMemory踩坑实录:重启丢数据、Agent丢记忆、对话溢出
后端·ai编程
壹方秘境9 小时前
我用Go语言开发了一个跨平台的HTTPS抓包和调试工具
前端·后端·ios
神秘面具男9 小时前
HarmonyOS 6.0跨端远程控制
前端·后端