Docker 中 RUN、CMD 与 ENTRYPOINT 的区别 2021-11-01 默认分类 暂无评论 250 次阅读 ### 前言 在说 `CMD、RUN 和 ENTRYPOINT` 的区别前,先来说说 `Dockerfile`,Dockerfile 是构建[容器镜像](https://so.csdn.net/so/search?q=%E5%AE%B9%E5%99%A8%E9%95%9C%E5%83%8F&spm=1001.2101.3001.7020)的方式之一,其通过一系列的指令参数来完成镜像的构建,而这些参数正是包含了 CMD,、RUN、COPY、ADD 和 ENTRYPOINT 等一系列指令。因此在实际应用中我们更多都是通过 Dockerfile 来完成镜像的构建。接下来列举一些 Dockerfile 常用的指令。 一、Dockerfile 常用指令 - FROM 指定基础(base)镜像,本地有镜像则直接使用,否则直接在线拉取(pull)。 - MAINTAINER Author,对作者的简单描述,自定义。 - COPY 将文件或目录从 build context [复制到](https://so.csdn.net/so/search?q=%E5%A4%8D%E5%88%B6%E5%88%B0&spm=1001.2101.3001.7020)镜像,其支持两种格式:COPY src dest 和 COPY\[“src”,“dest”\] > 注:原目标(src)只能是文件或目录。 - ADD 与 COPY 类似,[复制文件到](https://so.csdn.net/so/search?q=%E5%A4%8D%E5%88%B6%E6%96%87%E4%BB%B6%E5%88%B0&spm=1001.2101.3001.7020)镜像,不同的是,ADD 的 src 是归档文件(tar、zip、tgz 等),这些归档文件会被自动解压到 dest (镜像目标路径),无需手动解压。 - ENV 设置环境变量,该变量可被后面的指令使用。 - EXPOSE 指定容器中的进程会监听的某个端口,指定后 Docker 可以将该端口暴露出来。 - VOLUME 将文件或目录声明为 volume,同样 Docker 可以将该目录或文件映射出来。 - WORKDIR 为后面的 RUN、CMD、ENTRYPOINT、ADD、COPY 指令设置镜像中的当前工作目录。 - RUN 在容器中运行指令的命令。 - CMD 启动容器时运行指定的命令,Dockerfile 中可以有多个 CMD 指令,但只有最后一个生效,如果 docker run 后面指定有参数,该参数将会替换 CMD 的参数。 - ENTRYPOINT 同样,在 Dockerfile 中可以有多个 ENTRYPOINT 指令,也是只有最后一个生效,但与 CMD 不同的是,CMD 或 docker run 之后的参数会被当作参数传给 ENTRYPOINT。 二、三者的区别 2.1 Shell 和 Exec 格式 通常,我们有两种方式来指定 RUN、CMD 和 ENTRYPOINT 要运行的命令,即 Shell 和 Exec 方式。CMD 和 ENTRYPOINT 推荐使用 Exec 格式,其可读性更强。 1、Shell 格式 ```shell RUN yum install -y vim CMD echo "hello zhurs" ENTRYPOINT echo "hello zhurs" # 运行容器时返回如下结果 hello zhurs ``` 当指令执行时,Shell 格式会调用 `/bin/sh -c [command]`。 2、Exec 格式 > \[“executable”, “param1”, “param2”\] ```shell RUN ["yum", "install", "-y", "vim"] CMD ["bin/echo", "zhurs"] ENV wd world ENTRYPOINT ["/bin/echo", "hello, $wd"] # 运行容器时返回如下结果 hello $wd # 可看到运行容器时并没有调用/bin/sh -c 没有被shell解析(环境变量wd并没有被替换)。 # 如果希望使用环境变量,可做如下操作 RUN ["yum", "install", "-y", "vim"] CMD ["bin/echo", "zhurs"] ENV wd world ENTRYPOINT ["bin/sh", "-c", "/bin/echo", "hello, $wd"] # 此时就会返回如下结果 hello world ``` 2.2 RUN RUN 指令通常用于安装应用和软件包,每条 RUN 指令都会生成新的镜像。 ```shell ... RUN apt update && apt install -y git ... ``` 像在安装一些基础工具或应用的时候,apt update 和 apt install … 最好放在一个 RUN 指令下执行,因为这能够保证每次安装的是最新的包,如果 apt update 在单独的 RUN 下运行,则 apt install … 会使用 apt update 创建的镜像,而这一层镜像可能是很久以前缓存的镜像文件。 2.3 CMD 该指令用于用户启动容器时,容器来执行的命令,该命令会在容器启动且 docker run 后面没有指定其他命令时执行,所以小结三种情况: - docker run 没指定其他命令:则启动容器时运行 CMD 后的命令; - docker run 指定了其他命令:则启动容器时运行 CMD 后的命令会被忽略; - Dockerfile 中有多条 CMD 指令时,仅最后一条生效。 CMD 的三种格式: - shell 格式:CMD <二进制可执行命令> <指令1> <指令2> 如:CMD yum install -y vim - exec 格式:CMD \[“二进制可执行命令”, “指令1”, “指令2”\] 如:RUN \[“yum”, “install”, “-y”, “net-tools”\] - CMD \[“a”,“b”\] 格式:该格式是为 ENTRYPOINT 提供使用,此时 ENTRYPOINT 就必须使用 exec 格式,否则不生效。 2.4 ENTRYPOINT 该指令可以让容器以应用程序或者服务的形式运行。与 CMD 不同的是,不管 docker run … 后是否运行有其他命令,ENTRYPOINT 指令后的命令一定会被执行。 ENTRYPOINT 的两种格式: - shell 格式:同 CMD; - exec 格式:同 CMD。 ENTRYPOINT 的 exec 格式可以可执行由 CMD 提供的额外参数,具体如下: ```shell ... ENTRYPOINT ["/bin/echo", "hello"] CMD ["world"] ... ``` 运行容器时: - docker run -it > 运行的容器后无任何参数 ```shell # 输出 hello world ``` - docker run -it myworld > 运行的容器后跟了 myworld 参数 ```shell # 输出 hello myworld ``` **ENTRYPOINT 小结:** 无论运行的容器命令后是否有其他参数,ENTRYPOINT 一律执行,如果 ENTRYPOINT 后跟随有 CMD 指令参数,则该参数的内容将会作为 ENTRYPOINT 指令参数。如果 `docker run ...` 后有参数,ENTRYPOINT 则使用该参数,而不会使用 CMD 的参数,为什么呢?因为 `docker run ...` 后指定了参数,那么根据前面说到的 CMD 后的参数将会被忽略掉(或叫被替换)。 但是上面的结论是针对 ENTRYPOINT 的 exec 格式而言的,如果是 shell 格式,ENTRYPOINT 将会忽略掉任何 CMD 或 docker run … 提供的参数。当然 ENTRYPOINT 的 shell 格式也是会必然执行的。 2.5 如何选择 CMD 和 ENTRYPOINT > 根运行容器的属性来合理选择 - 如果运行的是一个 MySQL 容器,则优先使用 exec 格式的 ENTRYPOINT 指令,因为 CMD 不仅可以为 ENTRYPOINT 提供默认参数,同时在 `docker run ...`(带参数)的时候,该参数也会替换 CMD 默认参数。 - 如果只是简单的设置容器默认的启动命令,使用 CMD 即可,用户只需在 `docker run ...` 后添加参数即可替换默认值。 小结 - RUN:执行命令并创建新的镜像层,常用于安装软件包; - CMD:设置容器启动后默认执行的命令及其参数,但 docker run 后跟参数时会替换(忽略) CMD; - ENTRYPOINT:配置容器启动时运行的命令。 [转载自: https://blog.csdn.net/IT_ZRS/article/details/124707747](https://blog.csdn.net/IT_ZRS/article/details/124707747#start) 文章目录 一、Dockerfile 常用指令 二、三者的区别 2.1 Shell 和 Exec 格式 2.2 RUN 2.3 CMD 2.4 ENTRYPOINT 2.5 如何选择 CMD 和 ENTRYPOINT 小结 标签: docker 转载请注明文章来源 本作品采用 知识共享署名-相同方式共享 4.0 国际许可协议 进行许可。
评论已关闭