本篇文章是转载,原文

从Docker版本 17.05.0-ce 开始,就支持了一种新的构建镜像的方法,叫做:多阶段构建(Multi-stage builds),旨在解决Docker构建应用容器中的一些痛点。在日常构建容器的场景中,经常会遇到在同一个容器中进行源码的获取,编译和生成,最终才构建为镜像。这样做的劣势在于:

  • 不得不在容器中安装构建程序所必须的运行时环境
  • 不得不在同一个容器中,获取程序的源码和构建所需的一些生态工具
  • 构建出的镜像甚至包含了程序源码和一些不必要的文件,导致容器镜像尺寸偏大

当然,还有一种稍微优雅的方式,就是我们事先在外部将项目及其依赖库编译测试打包好后,再将其拷贝到构建目录中,这种虽然可以很好地规避第一种方式存在的风险点,但是也需要考虑不同镜像运行时,对于程序运行兼容性所带来的差异。

其实,这些痛点,Docker也想到了,官方提供了简便的多阶段构建 (multi-stage build) 方案。所谓多阶段构建,也即将构建过程分为多个阶段,在同一个Dockerfile中,通过不同的阶段来构建和生成所需要的应用文件,最终将这些应用文件添加到一个release的镜像中。这样做能完全规避上面所遇到的一系列问题。实现多阶段构建,主要依赖于新提供的关键字:fromas

下面举个栗子:

FROM muninn/glide:alpine AS build-env
ADD . /go/src/my-proj
WORKDIR /go/src/my-proj
RUN go get -v
RUN go build  -o /go/src/my-proj/my-serverFROM alpine
RUN apk add -U tzdata
RUN ln -sf /usr/share/zoneinfo/Asia/Shanghai  /etc/localtime
COPY --from=build-env /go/src/my-proj/my-server /my-server
EXPOSE 80
CMD ["my-server"]

多阶段构建的Dockerfile看起来像是把两个或者更多的Dockerfile合并在了一起,这也即多阶段的意思。as 关键字用来为构建阶段赋予一个别名,这样,在另外一个构建阶段中,可以通过 from 关键字来引用和使用对应关键字阶段的构建输出,并打包到容器中。

在多阶段构建完成之后,输出的镜像仅仅包含了最终输出的my-server应用,没有其他的源码文件和第三方源码包,非常的干净和简洁。因为 build-env 阶段只是一个构建的中间过程而已。

甚至,我们还可以使用更多的构建阶段来构建不同的应用,最终将这些构建产出的应用,合并到一个最终需要发布的镜像中。我们可以看一个更复杂一点的栗子:

from debian as build-essential
arg APT_MIRROR
run apt-get update
run apt-get install -y make gcc
workdir /srcfrom build-essential as foo
copy src1 .
run makefrom build-essential as bar
copy src2 .
run makefrom alpine
copy --from=foo bin1 .
copy --from=bar bin2 .
cmd ...

再来一个Laravel项目的多阶段构建( 自己加的内容)
第一阶段:使用compose安装PHP依赖
第二阶段:安装node,并安装前端依赖然后生成编译后的文件
第三阶段:拷贝PHP依赖及前端build后的文件到项目运行目录

#
# PHP Dependencies
#
FROM composer:1.7 as vendorCOPY database/ database/COPY composer.json composer.json
COPY composer.lock composer.lockRUN composer install \--ignore-platform-reqs \--no-interaction \--no-plugins \--no-scripts \--prefer-dist#
# Frontend
#
FROM node:8.11 as frontendRUN mkdir -p /app/publicCOPY package.json webpack.mix.js yarn.lock /app/
COPY resources/assets/ /app/resources/assets/WORKDIR /appRUN yarn install && yarn production#
# Application
#
FROM php:7.2-apache-stretchCOPY . /var/www/html
COPY --from=vendor /app/vendor/ /var/www/html/vendor/
COPY --from=frontend /app/public/js/ /var/www/html/public/js/
COPY --from=frontend /app/public/css/ /var/www/html/public/css/
COPY --from=frontend /app/mix-manifest.json /var/www/html/mix-manifest.json

多阶段构建的好处不言而喻,既可以很方便地将多个彼此依赖的项目通过一个Dockerfile就可轻松构建出期望的容器镜像,并且不用担心镜像太大、源码泄露等风险。不得不说,这是一个非常不错的改进。

参考:

https://docs.docker.com/develop/develop-images/multistage-build/
https://yq.aliyun.com/articles/181178
https://laravel-news.com/multi-stage-docker-builds-for-laravel

docker学习系列11 多阶段镜像构建相关推荐

  1. Docker学习笔记八:删除镜像构建私有Registry

    删除镜像 如果要删除本地的镜像,可以使用 docker rmi (注意rm为删除容器,而rmi为删除镜像,其中i代表image)命令,它的具体语法如下: docker rmi [OPTIONS] IM ...

  2. Docker学习系列从零开始之汇总【汇总】

    以下是整个docker学习系列的地址,希望可以给到大家一定的帮助: 1.Docker学习系列从零开始之centos6.5底下安装docker[一] 2.Docker学习系列从零开始之制作docker原 ...

  3. 狂神学习系列11:SpringBoot

    狂神学习系列11:SpringBoot 声明: 本文章是基于狂神的课程所编写,本人才疏学浅,内容仅作参考 项目和markdown文件资料: 07_SpringBoot: 基于狂神说SpringBoot ...

  4. Docker深入浅出系列 | 5分钟搭建镜像仓库

    Docker已经上市很多年,不是什么新鲜事物了,很多企业或者开发同学以前也不多不少有所接触,但是有实操经验的人不多,本系列教程主要偏重实战,尽量讲干货,会根据本人理解去做阐述,具体官方概念可以查阅官方 ...

  5. Docker学习笔记(一)-容器、镜像和客户端与守护进程

    0x00 前言 一直使用过docker,但并不是太了解,于是开始学习了解一下 0x01 docker简介 1.什么是容器? 一种虚拟化的方案 操作系统级别的虚拟化 只能运行相同或相似内核的操作系统 依 ...

  6. docker学习笔记(三)镜像

    参考:汤小洋老师的教学视频 docker学习笔记(一)[docker 介绍.卸载.安装.配置加速] docker学习笔记(二)docker常用命令 docker学习笔记(四)使用docker搭建环境 ...

  7. Docker学习(三):镜像

    2019独角兽企业重金招聘Python工程师标准>>> 1.简介 docker运行前需要本地存在对应的镜像,若镜像不存在本地,docker会先尝试从默认的镜像仓库下载(Docker ...

  8. Android学习系列(11)--App列表之拖拽ListView(下)

    接着上篇Android学习系列(10)--App列表之拖拽ListView(上)我们继续实现ListView的拖拽效果. 7.重写onTouchEvent()方法.      在这个方法中我们主要是处 ...

  9. 如何使用 Docker Buildx Bake 创建复杂的镜像构建管道

    docker buildx命令组使用BuildKit来公开高级图像构建功能.烘焙构建是一种高级功能,可用于定义自动构建管道.它们使您可以从单个构建操作中生成多个图像. 当您想要发布图像的不同变体或并行 ...

最新文章

  1. Nagios—服务和性能监控
  2. [微信开发] - 用户获取推广二维码
  3. ios 侧滑返回停顿_iOS push侧滑返回功能实现方法
  4. QT学习:视图(View)练习
  5. 系统运维岗位职责和要求
  6. 以太坊Dapp项目-网页钱包开发手册
  7. RTT 使用menuconfig 和cubeMux 在移植过来的模板工程上增加一个串口2
  8. 读《大道至简》第二章有感
  9. Android 资源(Resources)访问
  10. springboot日志写入mysql_springboot运用logback将日志写入数据库
  11. Head First SQL Your Brain on SQL读书笔记
  12. 微信小程序盲盒系统源码 带教程
  13. java http 传输二进制流_字符流、字节流、二进制及其在HTTP协议传输
  14. 纪念第一次面试安服-靶机测试过程以及思路
  15. 数据机房特殊规范标签生成及打印方案分享
  16. Bootstrap4模态框垂直居中
  17. 儿子于靖洋的百天照片
  18. 2022-2028年中国折叠椅行业发展策略分析及投资前景研究报告
  19. 大疆-2021招聘,我重点关注的岗位
  20. python将数据逐行写入excel_使用python将数据写入excel

热门文章

  1. Java HttpURLConnection示例– Java HTTP请求GET,POST
  2. java值的传递_Java 8可选–基于值的类使用指南
  3. Total Commander 常用快捷键
  4. 进程管理 systemctl控制
  5. 英国科研学术网络Janet遭遇DDoS攻击
  6. 设计模式解密(9)- 装饰者模式
  7. 支持Android的Qt5预览
  8. Using C++ in Eclipse - Program file not Specified problem
  9. 【整数转字符串】LeetCode 9. Palindrome Number
  10. 关于信道利用率的总结与一道习题的最终解释