使用Docker多阶段构建来减小镜像大小

 2023-12-24  阅读 2  评论 0

摘要:本文讲述了如何通过 的多阶段构建功能来大幅度减小镜像大小,适用于需要在 Dockerfile 中构建程式(如 javac),且需要另外安装编译工具链的镜像。(如 Java) 先来学习单词(本文全部采用中文词汇,如需查询外文文档可对照该词汇表。理论上个人不赞成翻译术语): multi-st

使用Docker多阶段构建来减小镜像大小

本文讲述了如何通过 的多阶段构建功能来大幅度减小镜像大小,适用于需要在 Dockerfile 中构建程式(如 javac),且需要另外安装编译工具链的镜像。(如 Java)

先来学习单词(本文全部采用中文词汇,如需查询外文文档可对照该词汇表。理论上个人不赞成翻译术语):

  • multi-stage 多阶段
  • build 构建
  • image 镜像
  • stage 阶段

再来看一下效果: 原 110M+,现 92M。

对比一下 Dockerfile

优化前 Dockerfile:

FROM openjdk:8u171-jdk-alpine3.8    ADD . /app  WORKDIR /app    RUN apk add maven     && mvn clean package     && apk del maven     && mv target/final.jar /     && cd /     && rm -rf /app     && rm -rf /root/.m2    ENTRYPOINT java -jar /final.jar

优化后 Dockerfile:

FROM openjdk:8u171-jdk-alpine3.8 as builder    ADD . /app  WORKDIR /app    RUN apk add maven     && mvn clean package     && apk del maven     && mv target/final.jar /    FROM openjdk:8u181-jre-alpine3.8 as environment  WORKDIR /  COPY --from=builder /final.jar .  ENTRYPOINT java -jar /final.jar    

很明显,优化后的 Dockerfile 新增了 FROM AS 这个命令,并出现了两个 FROM。这就是多阶段构建。

了解一下多阶段构建

多阶段构建是 Docker 17.05 的新增功能,它可以在一个 Dockerfile 中使用多个 FROM 语句,以创建多个 Stages(阶段)。每个阶段间独立(来源请求),可以通过 COPY –from 来获取其它阶段的文件。我们来打个比方,把最终镜像比作一盘菜(炒青椒)。把原料青椒炒完后上桌。

# 对比清单  镜像 -> 一盘菜  第一个阶段 -> 炒  第二个阶段 -> 上桌

两个阶段的目标是做好(生成)最终的菜(镜像)。我们要做的是将第一个阶段「炒」出来的食物进行「上桌」。我们的目标是 做出菜,且 菜盘子(盛菜和中间产物)最轻。

可视化流程如下:

# 做菜流程  ... 省略原料  原料 -> [第一个阶段——炒] # 此时盘子里有炒的工具、炒的结果和中间产物  # 这时候开启第二个阶段,只保留炒的结果,而不再需要其它。  -> 炒的结果 -> [开始上桌,只保留结果] # 把炒出来的青椒拿来(COPY --from),其它不要  -> 最终是一盘菜。

现在应该大致理解多阶段构建的流程了吧。我们把话筒交给 Java,看看在 Dockerfile 中使用编译工具构建一个 JAR,并只保留构建完的 JAR 和运行时交给 Image,其它则扔掉应该怎么做:

# 第一阶段——编译(炒)  FROM openjdk:8u171-jdk-alpine3.8 as builder # 自带编译工具    ADD . /app  WORKDIR /app    RUN ... 省略编译和清理工作...    # 现在,JAR 已经出炉。JDK 不再需要,所以不能留在镜像中。  # 所以我们开启第二阶段——运行(上桌),并扔掉第一阶段的所有文件(包括编译工具)  FROM openjdk:8u181-jre-alpine3.8 as environment # 只带运行时    # 目前,编译工具等上一阶段的东西已经被我们抛下。目前的镜像中只有运行时,我们需要把上一阶段(炒)的结果拿来,其它不要。  COPY --from=0 /final.jar .    # 好了,现在镜像只有必要的运行时和 JAR 了。  ENTRYPOINT java -jar /final.jar

如上就是多阶段构建的介绍。

使用多阶段构建

多阶段构建的核心命令是 FROM。FORM 对于身经百战的你来说已经不用多讲了。在多阶段构建中,每次 FROM 都会开启一个新的 Stage(阶段),可以看作一个新的 Image(不够准确、来源请求),与其它阶段隔离(甚至包括环境变量)。只有最后的 FROM 才会被纳入 Image 中。

我们来做一个最 simple 的多阶段构建例子:

# Stage 1  FROM alpine:3.8  WORKDIR /demo  RUN echo "Hello, stage 1" > /demo/hi-1.txt    # Stage 2  FROM alpine:3.8  WORKDIR /demo  RUN echo "Hello, stage 2" > /demo/hi-2.txt    

可以自己构建一下这个 Dockerfile,然后 docker save <tag> > docker.tar 看看其中的内容。不出意外应该只有 /demo/hi-2.txt 和 Alpine。

在这个 Dockerfile 中,我们创建了两个阶段。第一个阶段创建 hi-1.txt,第二个阶段创建 hi-2.txt,且第二个阶段会被加入最终 Image,其它不会。

复制文件——阶段间的桥梁

如果阶段间完全隔离,那么多阶段就没有意义——上一个阶段的结果会被完全抛弃,并进入全新的下一阶段。

我们可以通过 COPY 命令来获取其它阶段的文件。在多阶段中使用 COPY 和普通应用完全一致,仅需要添加 –form ` 即可。那么,我们修正上一个例子,使最终镜像包含两个阶段的产物:

# Stage 1  FROM alpine:3.8  WORKDIR /demo  RUN echo "Hello, stage 1" > /demo/hi-1.txt    # Stage 2  FROM alpine:3.8  WORKDIR /demo  COPY --from=0 /demo/hi-1.txt /demo  RUN echo "Hello, stage 2" > /demo/hi-2.txt    

重新构建并保存(Save),你会发现多了一层 Layer,其中包含 hi-1.txt。

阶段命名——快速识别

对于只有七秒记忆的我们来说,每次使用 stage index 并不是一件很妙的事情。这时候,可以通过阶段命名的方式给它们赋予名字,以方便识别。

为阶段添加名字很简单,只需要在 FROM 后加上 as <name> 即可。

现在,我们更新 Dockerfile,给予阶段名称并使用名称来 COPY。

# Stage 1, it's name is "build1"  FROM alpine:3.8 as build1  WORKDIR /demo  RUN echo "Hello, stage 1" > /demo/hi-1.txt    # Stage 2, it's name is "build2"  FROM alpine:3.8 as build2  WORKDIR /demo  # No longer use indexes  COPY --from=build1 /demo/hi-1.txt /demo  RUN echo "Hello, stage 2" > /demo/hi-2.txt    

重新构建并保存,结果应该同上次相同。

仅构建部分阶段——轻松调试

Docker 还为我们提供了一个很方便的调试方式——仅构建部分阶段。它可以使构建停在某个阶段,并不构建后面的阶段。这可以方便我们调试;区分生产、开发和测试。

仍然沿用上次的 Dockerfile,但使用 –target <stage> 参数进行构建:

$ docker build --target build1 .

再次 Save,你会发现只有 build1 的内容。

总结

这就是多阶段构建的全部用法了。我们再回到开篇的两个 Dockerfile 对比,你能发现优化前的镜像胖在哪里了吗?

很显然,它包含了无用的 JDK,JDK 只在编译时起作用,编译完便无用了,只需要 JRE 即可。所以,利用多阶段构建可以隔离编译阶段和运行阶段,以达到镜像最优化。

参考文献

https://docs.docker.com/develop/develop-images/multistage-build/#name-your-build-stages

https://yeasy.gitbooks.io/docker_practice/image/multistage-builds.html

提示:现在腾讯云新人点击注册然后实名认证后,可以点此一键领取2860元代金券,然后点此进入腾讯云活动页面参加优惠力度非常大的腾讯云3年和5年时长服务器活动,一次性买多年,免得续费贵,这样就可以获得最大的优惠折扣,省钱。

版权声明:xxxxxxxxx;

原文链接:https://lecms.nxtedu.cn/yunzhuji/130719.html

发表评论:

验证码

管理员

  • 内容1196554
  • 积分0
  • 金币0
关于我们
lecms主程序为免费提供使用,使用者不得将本系统应用于任何形式的非法用途,由此产生的一切法律风险,需由使用者自行承担,与本站和开发者无关。一旦使用lecms,表示您即承认您已阅读、理解并同意受此条款的约束,并遵守所有相应法律和法规。
联系方式
电话:
地址:广东省中山市
Email:admin@qq.com
注册登录
注册帐号
登录帐号

Copyright © 2022 LECMS Inc. 保留所有权利。 Powered by LECMS 3.0.3

页面耗时0.0132秒, 内存占用356.44 KB, 访问数据库18次

  • 我要关灯
    我要开灯
  • 客户电话
    lecms

    工作时间:8:00-18:00

    客服电话

    电子邮件

    admin@qq.com

  • 官方微信

    扫码二维码

    获取最新动态

  • 返回顶部