前言
现在越来越多的项目都向容器化考虑,云原生k8s注定成为未来的潮流。正好我们组里慢慢的从原生docker部署的方案慢慢迁移到Rancher部署,在一次技术分享里说到了Java项目打包大小的问题。
SpringBoot项目打包会包含我们自己的代码以及依赖的一些第三方jar包,一套技能打下来怎么也得100来M。SpringBoot官方在2.3.0以后支持了分层打包,今天浅试一波。
构建镜像
先随便创建个Springboot项目,如何编译成jar包以及Docker使用方式就不多BB了,可自行百度。做一个简单的Dockerfile测试。
pom配置默认打包插件
<build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin></plugins></build>
写一个非常简单的dockerfile
FROM openjdk:8 RUN mvn clean package -DskipTests # 声明端口并没有真正运行在这个端口EXPOSE8080ADD ./target/*.jar ./app.jar CMD java -jar app.jar
springboot项目就引入了一个spring-web包,然后写一个controller实现hello world测试即可。直接运行
docker build . -t docker-demo:0.2
我们直接使用docker inspect命令查看镜像构造,并查看分层结构
稍加改动一下项目重新走一遍打包,查看一下新镜像与之前做对比。
可以发现只有红框里一处编译与上一次不一样,也就是说我们只改动了几个字符,docker打包的时候依然从头加载了一遍springboot的全量jar包,如果仅仅是改动一点代码比如修复线上问题,这些工作显然是可以避免的。这也是导致打包速度慢的重要原因。
改造服务
首先要在默认的maven打包插件加入分层打包配置
<build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId><configuration><layers><enabled>true</enabled></layers></configuration></plugin></plugins></build>
再次进行打包分析操作mvn clean package,现在我们可以用下面命令来看分层打包编译的jar包结构
java-Djarmode=layertools-jartarget/dockers-demo-0.0.1-SNAPSHOT.jarlist
可以看到layertools识别出jar包内将依赖打包到不同文件夹中,接下来我们改造下原有的dockerfile。
改造dockerfile
先说下我们的适用场景,我们给前线部署之后可能会遇到一些bug修复,只会改一下部分代码,如果我们不分层打包的话,一次全量包会特别大效率非常低。
毕竟生产环境没有太大的bug改动,一般修复问题只需要几行代码,如果打全量包的话非常麻烦。
FROM openjdk:8 as builder RUN mvn clean package -DskipTests # 声明端口并没有真正运行在这个端口EXPOSE8080ADD ./target/*.jar ./app.jar RUN java -Djarmode=layertools -jar app.jar extract FROM openjdk:8-jre MAINTAINER ttzommed@foxmail.com WORKDIR application # 复制第三方依赖、SpringBoot内部配置、快照依赖COPY --from=builder application/dependencies/ ./ COPY --from=builder application/spring-boot-loader/ ./ COPY --from=builder application/snapshot-dependencies/ ./ COPY --from=builder application/application/ ./ EXPOSE8080ENTRYPOINT ["java", "org.springframework.boot.loader.JarLauncher"]
修改成这样就可以实现封层打包了~
总结
springboot官方设定2.3.0版本以上支持分层打包,并且dockerfile文件也改成相应的格式后,确实能达到我们预期的效果。
默认的分层为用户的代码,第三方依赖、SpringBoot内部配置、快照依赖 ,这些SpringBoot都为我们打包到不同的文件夹下。
依靠docker的分层特征,分次加入文件即可达到分层打包的效果。