云原生学习笔记(六) 一文学会使用 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 高效集成

相关推荐
容器魔方17 分钟前
科大讯飞基于Volcano实现AI基础设施突破,赢得CNCF最终用户案例研究竞赛
云原生·容器·云计算
程序员阿超的博客3 小时前
云原生核心技术 (10/12): K8s 终极实战:从零部署一个 Spring Boot + MySQL + Redis 应用
spring boot·云原生·kubernetes
东林牧之7 小时前
Docker搭建2FAuth服务
云原生·eureka
昌sit!7 小时前
K8S多维度问题排查
云原生·容器·kubernetes
kylin-运维7 小时前
k8s使用自建nfs做持久化无法控制磁盘使用大小问题处理
云原生·容器·kubernetes
movie__movie13 小时前
如何正确的配置eureka server集群
云原生·eureka
藥瓿亭15 小时前
K8S认证|CKS题库+答案| 11. AppArmor
运维·服务器·docker·云原生·容器·kubernetes·cks
955.15 小时前
k8s从入门到放弃之Ingress七层负载
云原生·容器·kubernetes
昌sit!19 小时前
K8S项目需求分析
云原生·容器·kubernetes
广州山泉婚姻21 小时前
高并发场景下的智慧零工平台开发:Spring Boot 3+MyBatis-Flex架构深度实践
分布式·爬虫·云原生