• Spring Boot layered(分层) jar 构建docker镜像


    这里是weihubeats,觉得文章不错可以关注公众号小奏技术,文章首发。拒绝营销号,拒绝标题党

    Docker分层

    Docker 容器由基本映像和附加层组成。一旦构建了各个层,它们将保持缓存状态,后续构建,就会使用缓存,构建更快
    在这里插入图片描述
    低层次的变化也会重建较高层次的变化。因此,我们在构建镜像的时候最好将不经常变化的图层保留在底部,而经常变化的图层应该放在顶部。这样就能提高构建docker效率和启动时间。

    Spring Boot分层

    Spring boot 在2.3之后也提供了应用分层,默认分层如下
    在这里插入图片描述
    这里可以看到应用层是独立一层,我们在修改应用代码的时候就只用构建应用层即可。spring-boot-loaderdependencies层就可以使用缓存,从而减少docker镜像的创建和启动时间。

    分层打包

    • 传统打包
    <build>
    		<plugins>
    			<plugin>
    				<groupId>org.apache.maven.pluginsgroupId>
    				<artifactId>maven-compiler-pluginartifactId>
    				<configuration>
    					<source>11source>
    					<target>11target>
    				configuration>
    			plugin>
    			<plugin>
    				<groupId>org.springframework.bootgroupId>
    				<artifactId>spring-boot-maven-pluginartifactId>
    				<version>${spring-boot.version}version>
    				<executions>
    					<execution>
    						<goals>
    							<goal>repackagegoal>
    						goals>
    					execution>
    				executions>
    			plugin>
    		plugins>
    	build>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

    这种方式是将整个Spring Boot打成一个jar,没有分层

    在Spring Boot 2.3之后添加了分层配置我们在打包插件中开启分层打包

    • 分层打包
    <build>
    		<plugins>
    			<plugin>
    				<groupId>org.apache.maven.pluginsgroupId>
    				<artifactId>maven-compiler-pluginartifactId>
    				<configuration>
    					<source>11source>
    					<target>11target>
    				configuration>
    			plugin>
    			<plugin>
    				<groupId>org.springframework.bootgroupId>
    				<artifactId>spring-boot-maven-pluginartifactId>
    				<version>${spring-boot.version}version>
    				<configuration>
    					<layers>
    						<enabled>trueenabled>
    					layers>
    				configuration>
    				<executions>
    					<execution>
    						<goals>
    							<goal>repackagegoal>
    						goals>
    					execution>
    				executions>
    			plugin>
    		plugins>
    	build>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29

    这样打出来的jar是分层的

    实战

    创建Spring Boot demo项目

    项目结构

    在这里插入图片描述

    pom 依赖
    
    <project xmlns="http://maven.apache.org/POM/4.0.0"
    		 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    		 xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    	<parent>
    		<artifactId>spirng-boot-kubernetesartifactId>
    		<groupId>com.zougroupId>
    		<version>1.0.0version>
    	parent>
    	<modelVersion>4.0.0modelVersion>
    
    	<artifactId>spring-boot-kuberntes-dockerartifactId>
    
    	<properties>
    		<maven.compiler.source>11maven.compiler.source>
    		<maven.compiler.target>11maven.compiler.target>
    	properties>
    
    	<dependencies>
    		<dependency>
    			<groupId>org.springframework.bootgroupId>
    			<artifactId>spring-boot-starter-webartifactId>
    		dependency>
    	dependencies>
    
    
    	<build>
    		<plugins>
    			<plugin>
    				<groupId>org.apache.maven.pluginsgroupId>
    				<artifactId>maven-compiler-pluginartifactId>
    				<configuration>
    					<source>11source>
    					<target>11target>
    				configuration>
    			plugin>
    			<plugin>
    				<groupId>org.springframework.bootgroupId>
    				<artifactId>spring-boot-maven-pluginartifactId>
    				<version>${spring-boot.version}version>
    				<configuration>
    					<layers>
    						<enabled>trueenabled>
    					layers>
    				configuration>
    				<executions>
    					<execution>
    						<goals>
    							<goal>repackagegoal>
    						goals>
    					execution>
    				executions>
    			plugin>
    		plugins>
    	build>
    
    project>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    测试代码
    • Application
    @SpringBootApplication
    public class Application {
    
    	public static void main(String[] args) {
    		TimeZone.setDefault(TimeZone.getTimeZone("Asia/Shanghai"));
    		SpringApplication.run(Application.class, args);
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • TestController
    @RestController
    @RequestMapping
    public class TestController {
    
    	@GetMapping("test")
    	public String test() {
    		return "hello docker image";
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    编写Dockerfile
    # 指定基础镜像,这是分阶段构建的前期阶段
    FROM adoptopenjdk/openjdk11 as builder
    # 执行工作目录
    WORKDIR application
    # 配置参数
    ARG JAR_FILE=target/*.jar
    # 将编译构建得到的jar文件复制到镜像空间中
    COPY ${JAR_FILE} application.jar
    # 通过工具spring-boot-jarmode-layertools从application.jar中提取拆分后的构建结果
    RUN java -Djarmode=layertools -jar application.jar extract
    
    # 正式构建镜像
    FROM adoptopenjdk/openjdk11
    WORKDIR application
    # 前一阶段从jar中提取除了多个文件,这里分别执行COPY命令复制到镜像空间中,每次COPY都是一个layer
    COPY --from=builder application/dependencies/ ./
    COPY --from=builder application/spring-boot-loader/ ./
    COPY --from=builder application/snapshot-dependencies/ ./
    COPY --from=builder application/application/ ./
    ENTRYPOINT ["java", "org.springframework.boot.loader.JarLauncher"]
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    打包

    我们在项目根目录执行打包命令

    mvn clean package -U -DskipTests
    
    • 1

    如果我们解压我们的jar就会发现多了layers.idx文件,打开文件我们会发现有如下内容

    dependencies
    spring-boot-loader
    snapshot-dependencies
    application
    
    • 1
    • 2
    • 3
    • 4
    构建docker 镜像

    我们在Dockerfile根目录执行如下命令

    docker build -t dockerlayerdemo:0.0.5 .
    
    • 1

    在这里插入图片描述

    第一次构建可能会有点慢,如果我们后续改了文件重新打包成镜像会快很多,因为有缓存,我们打包又是分了层,这种分层方式比打整个jar包的方式大概能快1倍的速度左右,具体性能差异还是要自己多测试

    运行docker镜像

    执行如下命令

    docker run -d -p 8080:8080 dockerlayerdemo:0.0.5
    
    • 1

    启动完成后我们访问测试接口

    localhost:8080/test
    
    • 1

    可以看到可以正常访问
    在这里插入图片描述

    自定义分层

    如果我们仔细看上面的会发现依赖不会在构建的时候共享,所以我们需要自定义配置调整下
    在Spring Boot中我们可以通过配置文件来调优自定义分层
    我们建立一个layers.xml

    <layers xmlns="http://www.springframework.org/schema/boot/layers"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://www.springframework.org/schema/boot/layers
                         https://www.springframework.org/schema/boot/layers/layers-2.3.xsd">
        <application>
            <into layer="spring-boot-loader">
                <include>org/springframework/boot/loader/**include>
            into>
            <into layer="application" />
        application>
        <dependencies>
            <into layer="snapshot-dependencies">
                <include>*:*:*SNAPSHOTinclude>
            into>
            <into layer="dependencies" />
        dependencies>
        <layerOrder>
            <layer>dependencieslayer>
            <layer>spring-boot-loaderlayer>
            <layer>snapshot-dependencieslayer>
            <layer>applicationlayer>
        layerOrder>
    layers>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    然后再maven中指定配置文件

    <plugin>
        <groupId>org.springframework.bootgroupId>
        <artifactId>spring-boot-maven-pluginartifactId>
        <configuration>
            <layers>
                <enabled>trueenabled>
                <configuration>${project.basedir}/src/layers.xmlconfiguration>
            layers>
        configuration>
    plugin>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    这样我们可以自定义分层了

    总结

    总的来说要构建高效的Docker镜像并不是那么简单,也是有很多学问要研究的,后续有机会继续深入研究下

    参考

  • 相关阅读:
    【LVGL事件(Events)】事件代码
    23年秋招笔试算法题:美团蔚来
    无线设备的天线怎么安装最好?
    java的Integer中也会有缓存
    部门来了个测试开发,听说是00后,上来一顿操作给我看呆了...
    超标量处理器设计 姚永斌 第3章 虚拟存储器 --3.3~3.4 小节摘录
    ZZ308 物联网应用与服务赛题第A套
    28335之GPIO输入
    回溯算法 | 分割字符串 | 复原IP地址 | leecode刷题笔记
    力扣(LeetCode)813. 最大平均值和的分组(C++)
  • 原文地址:https://blog.csdn.net/qq_42651904/article/details/126391793