云原生学习笔记(六) 一文学会使用 Dockerfile:构建镜像的黄金剧本

📓 前言

Dockerfile 是构建 Docker 镜像的脚本文件,定义了镜像内应该包含的内容和构建步骤。掌握它,就等于掌握了容器化应用的核心生产力工具。

参考文档


🧱 一、Dockerfile 是什么?

Dockerfile 是一个文本文件 ,包含了一系列用于构建镜像的指令(Instructions)。这些指令会被 docker build 命令顺序执行,最终生成一个可运行的容器镜像。


🧩 二、Dockerfile 核心指令全览

一个标准的 Dockerfile 通常包括以下几个核心部分:

指令 作用
FROM 指定基础镜像(如 ubuntu, alpine, python
RUN 在构建镜像时运行命令(如安装依赖)
COPY / ADD 将本地文件复制到镜像中
WORKDIR 设置工作目录
CMD / ENTRYPOINT 设置容器启动时执行的命令
EXPOSE 声明容器监听的端口(文档用途)
ENV 设置环境变量
ARG 定义构建时变量
VOLUME 定义挂载卷位置(持久化数据)
LABEL 添加元数据信息(如作者、版本)

以下是构建镜像最常用的 Dockerfile 指令:

1. FROM(基础镜像)

定义构建镜像所基于的父镜像。必须是第一条非注释指令。

dockerfile 复制代码
FROM node:20-alpine

2. RUN(执行命令)

在镜像构建时执行 shell 命令,常用于安装依赖、修改配置。

dockerfile 复制代码
RUN apk add --no-cache python3

3. COPY / ADD(复制文件)

将宿主机上的文件复制到镜像中。

  • COPY:基础复制
  • ADD:支持远程 URL、自动解压 tar 文件
dockerfile 复制代码
COPY . /app
ADD https://example.com/data.tar.gz /data/

4. WORKDIR(设置工作目录)

设置后续指令的默认工作路径。

dockerfile 复制代码
WORKDIR /app

5. CMD / ENTRYPOINT(定义启动命令)

  • CMD:容器默认启动命令,可被覆盖
  • ENTRYPOINT:更强制的启动命令(不容易被替换)
dockerfile 复制代码
CMD ["node", "index.js"]
ENTRYPOINT ["python3", "main.py"]

🔎 CMD vs ENTRYPOINT 有啥区别?

项目 CMD ENTRYPOINT
默认命令
可被 docker run 覆盖 ❌(除非加 --entrypoint
最常用途 提供默认命令参数 设置主命令(如 nginx

结合使用:

dockerfile 复制代码
ENTRYPOINT ["python"]
CMD ["app.py"]


6. EXPOSE(声明端口)

声明容器运行时监听的端口(仅声明,不会自动映射主机端口)。

dockerfile 复制代码
EXPOSE 8080

7. ENV(设置环境变量)

dockerfile 复制代码
ENV NODE_ENV=production

8. ARG(构建参数)

在构建阶段可传递变量,通常与 --build-arg 配合使用。

dockerfile 复制代码
ARG VERSION=1.0
RUN echo "Build version: $VERSION"

🧠 三、Dockerfile 编写建议

  1. 选择轻量镜像 :如 alpine,减小体积、提升构建速度。

  2. 减少层数 :将多个 RUN 命令合并。

    dockerfile 复制代码
    RUN apt update && apt install -y curl && rm -rf /var/lib/apt/lists/*
  3. 使用 .dockerignore:避免将无关文件复制进镜像。

  4. 固定版本号:确保构建的可重复性。

  5. 使用 ENTRYPOINT + CMD 组合:提高灵活性。

  6. 利用缓存优化构建顺序:常变的文件放后面。

  7. 多阶段构建(Multi-stage build): 只保留运行时必要内容,精简镜像体积。


🧪 四、示例:打包一个 Python Flask 应用

dockerfile 复制代码
# 使用官方 Python 基础镜像
FROM python:3.10-slim

# 设置工作目录
WORKDIR /app

# 复制依赖文件并安装
COPY requirements.txt .
RUN pip install -r requirements.txt

# 复制源代码
COPY . .

# 暴露端口
EXPOSE 5000

# 启动应用
CMD ["python", "app.py"]

构建镜像:

bash 复制代码
docker build -t flask-demo .

运行容器:

bash 复制代码
docker run -p 5000:5000 flask-demo

🚫 五、常见坑点

  • COPY 顺序错 → 构建缓存失效
  • RUN 安装完没清理缓存 → 镜像超大
  • WORKDIR 未设置 → 程序路径混乱
  • EXPOSE 没映射 → 容器内部服务无法访问
  • ENTRYPOINT 写死 → 无法调试或扩展

🚀 六、总结

Dockerfile 是构建镜像的描述性语言,掌握它可以:

  • 标准化部署流程
  • 避免"在我电脑上可以跑"
  • 与 CI/CD 高效集成

相关推荐
努力搬砖的咸鱼12 小时前
Docker 三剑客:镜像、容器、仓库
docker·云原生·容器
稚辉君.MCA_P8_Java15 小时前
View:new关键词干了什么事,还有原型链是什么
后端·云原生
lingggggaaaa15 小时前
小迪安全v2023学习笔记(九十七天)—— 云原生篇&Kubernetes&K8s安全&API&Kubelet未授权访问&容器执行
java·笔记·学习·安全·网络安全·云原生·kubernetes
做运维的阿瑞17 小时前
GPU即服务:Linux与云原生如何联手开启AI算力“自来水“时代
linux·人工智能·云原生
hzulwy18 小时前
微服务注册与监听
微服务·云原生·架构·go
yuezhilangniao21 小时前
基础架构安全和云原生安全的融合~K8S安全和传统安全~K8S和安全融合~综合安全大饼
安全·云原生·kubernetes
nvd111 天前
用terraform 创建一个GKE private cluster
云原生·kubernetes·terraform·gke
Q飞了1 天前
深入理解k8s中pod、service、deployment和statefulSet等工作负载--图文篇
云原生·容器·kubernetes
NightReader1 天前
minikube 的 kubernetes 入门教程-kubeSphere
云原生·容器·kubernetes
10岁的博客2 天前
无Dockerfile构建:云原生部署新姿势
云原生