Docker镜像优化:从1.16GB到22.4MB

运维 系统运维
在这篇文章中,我将重点讨论如何优化 Docker 镜像以使其轻量化。

 Docker 是一个供软件开发人员和系统管理员使用容器构建、运行和与分享应用程序的平台。容器是在独立环境中运行的进程,它运行在自己的文件系统上,该文件系统是使用 docker 镜像构建的。镜像中包含运行应用程序所需的一切(编译后的代码、依赖项、库等等)。镜像使用 Dockerfile 文件定义。

[[378374]]

术语 dockerization 或 containerization 通常用于定义创建 Docker 容器的过程。

因为容器具备如下优点,所以很受欢迎:

  •  灵活性:即使是最复杂的应用程序也可以容器化。
  •  轻量化:容器共享主机内核,使得它们远比虚拟机高效。
  •  便携性:可以做到本地编译,到处运行。
  •  松耦合:容器自我封装,一个容器被替换或升级不会打断别的容器。
  •  安全性:容器对进程进行了严格的限制和隔离,而无需用户进行任何配置。

在这篇文章中,我将重点讨论如何优化 Docker 镜像以使其轻量化。

让我们从一个示例开始,在该示例中,我们构建了一个 React 应用程序并将其容器化。运行 npx 命令并创建 Dockerfile 之后,我们得到了如图 1 所示的文件结构。 

  1. npx create-react-app app --template typescript 

图 1:文件结构

如果我们构建一个基础的 Dockerfile(如下所示),我们最终会得到一个 1.16 GB 的镜像: 

  1. FROM node:10  
  2. WORKDIR /app  
  3. COPY app /app  
  4. RUN npm install -g webserver.local  
  5. RUN npm install && npm run build  
  6. EXPOSE 3000  
  7. CMD webserver.local -d ./build 

图 2:镜像的初始大小为 1.16GB

第一步优化:使用轻量化基础镜像

在 Docker Hub(公共 Docker 仓库)中,有一些镜像可供下载,每个镜像都有不同的特征和大小。

通常,相较于基于其他 Linux 发行版(例如 Ubuntu)的镜像,基于 Alpine 或 BusyBox 的镜像非常小。这是因为 Alpine 镜像和类似的其他镜像都经过了优化,其中仅包含最少的必须的软件包。在下面的图片中,你可以看到 Ubuntu、Alpine、Node 和基于 Alpine 的 Node 镜像之间的大小比较。

图 3:基础镜像的不同大小

通过修改 Dockerfile 并使用 Alpine 作为基础镜像,我们的镜像最终大小为 330MB: 

  1. FROM node:10-alpine  
  2. WORKDIR /app  
  3. COPY app /app  
  4. RUN npm install -g webserver.local  
  5. RUN npm install && npm run build  
  6. EXPOSE 3000  
  7. CMD webserver.local -d ./build 

图 4:经过第一步优化后镜像大小为 330MB

第二步优化:多阶段构建

通过多阶段构建,我们可以在 Dockerfile 中使用多个基础镜像,并将编译成品、配置文件等从一个阶段复制到另一个阶段,这样我们就可以丢弃不需要的东西。

在本例中,我们部署 React 应用程序需要的是编译后的代码,我们不需要源文件,也不需要 node_modules 目录和 package.json 文件等。

通过将 Dockerfile 修改为如下内容,我们最终得到的镜像大小为 91.5MB。请记住,来自第一阶段(第 1-4 行)的镜像不会被自动删除,Docker 将它保存在 cache 中,如果我们在另一个构建镜像过程中执行了相同的阶段,就可以使镜像构建更快。所以你必须手动删除第一阶段镜像。 

  1. FROM node:10-alpine AS build  
  2. WORKDIR /app  
  3. COPY app /app  
  4. RUN npm install && npm run build  
  5. FROM node:10-alpine  
  6. WORKDIR /app  
  7. RUN npm install -g webserver.local  
  8. COPY --from=build /app/build ./build  
  9. EXPOSE 3000  
  10. CMD webserver.local -d ./build 

图 5:第二步优化后的镜像大小为 91.5MB

现在我们有了一个 Dockerfile,它有两个阶段:在第一个阶段中,我们编译项目,在第二个阶段中,我们在 web 服务器上部署应用程序。然而,Node 容器并不是提供网页(HTML、CSS 和 JavaScript 文件、图片等)服务的最佳选择,最好的选择是使用像 Nginx 或 Apache 这样的服务。在本例中,我将使用 Nginx。

通过将 Dockerfile 修改为如下内容,我们的镜像最终大小是 22.4MB,如果我们运行这个容器,我们可以看到网页可以正常工作,没有任何问题(图 7)。 

  1. FROM node:10-alpine AS build  
  2. WORKDIR /app  
  3. COPY app /app  
  4. RUN npm install && npm run build  
  5. FROM nginx:stable-alpine  
  6. COPY --from=build /app/build /usr/share/nginx/html  
  7. EXPOSE 80  
  8. CMD ["nginx", "-g", "daemon off;"] 

图 6:第三步优化后的镜像大小为 22.4MB

图 7:最终容器的运行结果 

 

责任编辑:庞桂玉 来源: 运维派
相关推荐

2021-08-30 07:08:59

Docker镜像服务器

2022-05-02 17:03:32

容器镜像ReactJS

2024-10-31 10:41:01

2017-02-09 17:30:05

Android应用瘦身

2021-04-27 08:53:37

容器镜像

2020-08-23 11:52:10

Docker容器技术

2021-05-13 23:54:12

DockerDockerfile镜像

2012-03-08 21:41:10

App Store

2011-11-17 11:24:37

CA World 20CA Technolo

2009-09-08 18:29:43

敏捷开发

2024-01-29 08:24:40

2024-10-17 09:57:30

2022-05-12 08:04:50

Docker镜像Python

2017-01-09 09:34:03

Docker容器传统虚拟机

2024-01-19 16:35:23

2017-07-18 11:12:39

环境设置内存分析Python

2020-03-19 10:44:19

DockerSpring Boo单层镜像

2020-11-16 09:05:08

Docker架构容器

2024-12-02 11:24:30

Docker编排技术

2023-08-28 16:10:00

容器化DockerKubernetes
点赞
收藏

51CTO技术栈公众号