Dockerfile中run、cmd和entrypoint都能够用于执行命令,下面是三者的主要用途:

  • run命令执行命令并创建新的镜像层,通常用于安装软件包
  • cmd命令设置容器启动后默认执行的命令及其参数,但CMD设置的命令能够被docker run命令后面的命令行参数替换
  • entrypoint配置容器启动时的执行命令,不会被忽略,一定会被执行,即使运行 docker run时指定了其他命令。

Shell格式和Exec格式运行命令

我们可以用下面两种格式指定run、cmd和entrypoint要运行的命令:

  • Shell格式: 。例如:yum install -y wget
  • Exec格式: [“executable”, “param1”, “param2”, …]。例如: [“yum”, “install”, “-y”, “wget”]

cmd和entrypoint推荐使用exec格式,因为指令的可读性更强,更容易理解,而run则两种格式都可以。

Exec格式的坑

dockerfile的内容如下:

env name morris

entrypoint ["echo", "$name"]

这种写法只会打印出$name,不会进行变量的替换,原因是它只是在执行echo命令,并不是执行shell。意思是说,我们不是在shell里执行echo,只是单纯的执行echo,所以不会替换变量。

想要改成可执行的shell,需要改写成以下形式

env name morris

entrypoint ["/bin/bash", "-c", "echo $name"]

run命令

run指令通常用于安装应用和软件包。run在当前镜像的顶部执行命令,并通过创建新的镜像层。Dockerfile中常常包含多个run指令。下面是一个例子:

run yum update && yum install -y \  
 bzr \
 cvs \
 git \
 mercurial \
 subversion

yum update和yum install被放在一个run指令中执行,这样能够保证每次安装的是最新的包。如果yum install在单独的run中执行,则会使用yum update创建的镜像层,而这一层可能是很久以前缓存的。

cmd命令

cmd指令允许用户指定容器的默认执行的命令。此命令会在容器启动且docker run没有指定其他命令时运行。下面是一个例子:

cmd echo "Hello world"

运行容器docker run -it [image]将输出:

Hello world

但当后面加上一个命令,比如docker run -it [image] echo hi,cmd会被忽略掉,命令echo hi将被执行:

hi

如果存在多个cmd命令,则只会执行最后一个cmd命令。

entrypoint命令

entrypoint的exec格式用于设置容器启动时要执行的命令及其参数,同时可通过cmd命令或者命令行参数提供额外的参数。entrypoint中的参数始终会被使用,这是与cmd命令不同的一点。下面是一个例子:

entrypoint ["echo", "Hello"]

当容器通过docker run -it [image]启动时,输出为:

Hello

而如果通过docker run -it [image] morris启动,则输出为:

Hello morris

再来看一个例子,Dockerfile为:

entrypoint ["echo", "Hello"]
cmd ["world"]

当容器通过docker run -it [image]启动时,输出为:

Hello world

而如果通过docker run -it [image] morris启动时,输出为:

Hello morris

entrypoint中的参数始终会被使用,而cmd的额外参数可以在容器启动时动态替换掉。

同样的,如果存在多个entrypoint命令,则只会执行最后一个entrypoint命令。

总结

  • 使用run指令安装应用和软件包,构建镜像。
  • 如果Docker镜像的用途是运行应用程序或服务,比如运行一个MySQL,应该优先使用Exec格式的entrypoint指令。cmd可为entrypoint提供额外的默认参数,同时可利用docker run命令行替换默认参数。
  • 如果想为容器设置默认的启动命令,可使用cmd指令。用户可在docker run命令行中替换此默认命令。