代码编织梦想

环境

  • RHEL 9.3
  • Docker Community 24.0.7

总结

如果懒得看详细介绍,可以直接看总结:

  • ENTRYPOINTCMD 都可以单独使用,指定启动容器时所运行的命令以及参数。
  • 更常见的用法是把 ENTRYPOINTCMD 组合使用:
    • ENTRYPOINT 指定启动容器时所运行的命令和不变的参数。在启动容器时可以显式覆盖,但一般不这么做。
    • CMD 指定运行参数。在启动容器时可以显式覆盖。
  • ENTRYPOINTCMD 都强烈推荐使用“exec形式”。

例如:

ENTRYPOINT ["ping", "-c", "20"]
CMD ["localhost"]

讲解

ENTRYPOINTCMD 指定启动容器时所运行的命令以及参数。二者都可以单独使用,也可以把二者组合使用。接下来通过示例来看一下二者的用法。

(一)不指定ENTRYPOINT和CMD

创建 Dockerfile 文件如下:

FROM ubuntu:trusty

构建:

docker build -t kaidemo0 .

启动容器:

docker run kaidemo0

结果什么也没有发生:既没有报错,也没有任何输出。

通过 docker ps -a 查看容器:

➜  ~ docker ps -a
CONTAINER ID   IMAGE                    COMMAND                  CREATED          STATUS                      PORTS                      NAMES
f3aeb967fdd2   kaidemo0                 "/bin/bash"              49 seconds ago   Exited (0) 48 seconds ago                              affectionate_noyce

可以看到,在没有指定 ENTRYPOINTCMD 时,实际运行的命令是 /bin/bash

要想与容器交互,可以加上 -it 选项:

➜  ~ docker run -it kaidemo0
root@1389cac2f57e:/# ls
bin  boot  dev  etc  home  lib  lib64  media  mnt  opt  proc  root  run  sbin  srv  sys  tmp  usr  var
root@1389cac2f57e:/# ps -ef
UID          PID    PPID  C STIME TTY          TIME CMD
root           1       0  0 09:07 pts/0    00:00:00 /bin/bash
root          18       1  0 09:07 pts/0    00:00:00 ps -ef
root@1389cac2f57e:/# exit
exit

注:

  • -i :interactive,交互式的,接收用户输入
  • -t :tty,分配一个伪终端

也可在 docker run 时指定运行的命令,比如:

➜  ~ docker run kaidemo0 ping localhost
PING localhost (127.0.0.1) 56(84) bytes of data.
64 bytes from localhost (127.0.0.1): icmp_seq=1 ttl=64 time=0.069 ms
64 bytes from localhost (127.0.0.1): icmp_seq=2 ttl=64 time=0.032 ms
64 bytes from localhost (127.0.0.1): icmp_seq=3 ttl=64 time=0.024 ms

在另一个命令行窗口,查看容器:

➜  test1 docker ps
CONTAINER ID   IMAGE            COMMAND                  CREATED         STATUS         PORTS                      NAMES
ab8ed71eae08   kaidemo0         "ping localhost"         3 seconds ago   Up 3 seconds                              interesting_dijkstra

最后,在第一个窗口,按下“Ctrl + C”停止ping命令,同时退出容器。

总结:若不指定ENTRYPOINT和CMD,启动容器时,默认运行的命令是 /bin/bash 。要显式加上 -it 选项才能进入容器做事。也可以在启动容器时显式指定运行的命令。

(二)CMD

创建 Dockerfile 文件如下:

FROM ubuntu:trusty
CMD ping localhost

构建:

docker build -t kaidemo1 .

启动容器:

docker run kaidemo1

由于 CMD 指令指定了 ping localhost 命令,容器启动时会自动运行该命令,如下:

➜  ~ docker run kaidemo1      
PING localhost (127.0.0.1) 56(84) bytes of data.
64 bytes from localhost (127.0.0.1): icmp_seq=1 ttl=64 time=0.145 ms
64 bytes from localhost (127.0.0.1): icmp_seq=2 ttl=64 time=0.199 ms
64 bytes from localhost (127.0.0.1): icmp_seq=3 ttl=64 time=0.065 ms
^C64 bytes from localhost (127.0.0.1): icmp_seq=4 ttl=64 time=0.084 ms
64 bytes from localhost (127.0.0.1): icmp_seq=5 ttl=64 time=0.030 ms
......

注意:按下 “Ctrl + C” 无法停止ping命令,原因稍后解释。

在另一个命令行窗口,查看docker容器:

➜  ~ docker ps
CONTAINER ID   IMAGE            COMMAND                  CREATED         STATUS         PORTS                      NAMES
25e553984c8a   kaidemo1         "/bin/sh -c 'ping lo…"   7 seconds ago   Up 7 seconds                              sleepy_lalande

同样,尝试停止容器也无效:

➜  ~ docker stop 25e553984c8a
^C

docker stop 命令hang住了,只能按“Ctrl + C”中止。

最后,运行 docker rm -f 强制删除容器:

➜  ~ docker rm -f 25e553984c8a
25e553984c8a

原因解释:通过刚才的 docker ps 可以看到,实际运行的命令是 /bin/sh -c 'ping localhost'

  • docker run kaidemo0 ping localhost 时,在容器里查看进程:
➜  docker docker exec <container ID> ps -ef       
UID          PID    PPID  C STIME TTY          TIME CMD
root           1       0  0 02:08 ?        00:00:00 ping localhost
root           7       0  0 02:09 ?        00:00:00 ps -ef
  • docker run kaidemo1 时,在容器里查看进程:
➜  docker docker exec <container ID> ps -ef
UID          PID    PPID  C STIME TTY          TIME CMD
root           1       0  0 02:12 ?        00:00:00 /bin/sh -c ping localhost
root           7       1  0 02:12 ?        00:00:00 ping localhost
root           8       0  0 02:12 ?        00:00:00 ps -ef
  • docker run -it kaidemo1 时,在容器里查看进程:
➜  docker docker exec <container ID> ps -ef
UID          PID    PPID  C STIME TTY          TIME CMD
root           1       0  0 02:16 pts/0    00:00:00 /bin/sh -c ping localhost
root           8       1  0 02:16 pts/0    00:00:00 ping localhost
root           9       0  0 02:16 ?        00:00:00 ps -ef

注意PID为1的进程,分别为:

  • ping localhost
  • /bin/sh -c ping localhost (没有TTY)
  • /bin/sh -c ping localhost (有TTY)

当我们在容器外部发送POSIX信号(比如“Ctrl + C”)到容器里,对于 docker run kaidemo1/bin/sh 命令不会转发消息给实际运行的ping命令,所以无法停止ping。

之所以出现这样的问题,是因为我们在Dockerfile里使用了“shell形式”,即:

CMD ping localhost

Docker会把该命令作为shell的子命令(即 /bin/sh -c xxxxx ),这就带来了问题。

为了避免这个问题,可以使用“exec形式”。

创建 Dockerfile 文件如下:

FROM ubuntu:trusty
CMD ["ping", "localhost"]

构建:

docker build -t kaidemo2 .

启动容器:

docker run kaidemo2

在另一个命令行窗口,查看容器:

➜  docker docker ps
CONTAINER ID   IMAGE            COMMAND                  CREATED         STATUS         PORTS                      NAMES
a6e67096ca3a   kaidemo2         "ping localhost"         6 seconds ago   Up 4 seconds                              stupefied_euler

查看容器里的进程:

➜  ~ docker exec <container ID> ps -ef
UID          PID    PPID  C STIME TTY          TIME CMD
root           1       0  0 02:33 ?        00:00:00 ping localhost
root           7       0  0 02:33 ?        00:00:00 ps -ef

可见,应尽量使用“exec形式”,以避免子shell的问题。

可以在启动容器时,显式指定要运行的命令,覆盖 CMD 的命令。例如:

➜  ~ docker run kaidemo2 ls -l 
total 8
drwxr-xr-x.   2 root root 4096 Dec 17  2019 bin
drwxr-xr-x.   2 root root    6 Apr 10  2014 boot
drwxr-xr-x.   5 root root  340 Jan  6 02:39 dev
......

总结:假设没有指定 ENTRYPOINT 指令,则 CMD 指令可以指定启动启动时要运行的命令。强烈推荐使用“exec形式”,以避免子shell的问题。可以在启动容器时,显式指定要运行的命令,覆盖 CMD 指定的命令。

(三)ENTRYPOINT

创建 Dockerfile 文件如下:

FROM ubuntu:trusty
ENTRYPOINT ping localhost

构建:

docker build -t kaidemo3 .

启动容器,可以发现, ENTRYPOINTCMD 的表现几乎一模一样。

因此,应尽量使用“exec形式”:

ENTRYPOINT ["ping", "localhost"]

另外,在启动容器时,覆盖 ENTRYPOINT 的方法和 CMD 不同,例如:

➜  ~ docker run kaidemo4 ls       
ping: unknown host ls

可见,运行的还是ping命令,只不过参数变成了 ls 。这是因为覆盖的是 CMD 指令,而不是 ENTRYPOINT 指令。关于二者的组合,稍后会有介绍。

要想覆盖 ENTRYPOINT 指令,可以使用 --entrypoint 选项:

➜  ~ docker run --entrypoint ls kaidemo4 
bin
boot
dev
......

但是,不推荐使用这种做法,原因稍后会有介绍。

总结: ENTRYPOINT 的表现几乎和 CMD 完全一致。假设没有指定 CMD 指令,则 ENTRYPOINT 指令可以指定启动启动时要运行的命令。强烈推荐使用“exec形式”,以避免子shell的问题。可以在启动容器时,通过 --entrypoint 选项显式指定要运行的命令,覆盖 ENTRYPOINT 指定的命令,但一般不这么做。

(四)ENTRYPOINT和CMD的组合

前面说了这么多, ENTRYPOINTCMD 貌似也没什么本质的区别,那为什么Dockerfile里要有两个相似的指令呢?

实际上,二者的设计理念不一样,典型的用法是把它们组合起来使用:

  • ENTRYPOINT :指定默认的启动程序
  • CMD :指定默认的运行参数

创建 Dockerfile 文件如下:

FROM ubuntu:trusty
ENTRYPOINT ["ping", "-c", "20"]
CMD ["localhost"]

注: -cping 命令的选项,c表示count, -c 20 就是ping 20次。

构建:

docker build -t kaidemo5 .

启动容器:

docker run kaidemo5

在另一个命令行窗口,查看容器:

➜  ~ docker ps                      
CONTAINER ID   IMAGE            COMMAND                  CREATED         STATUS         PORTS                      NAMES
135c6466f888   kaidemo5         "ping -c 20 localhost"   3 seconds ago   Up 2 seconds                              upbeat_vaughan

查看容器里的进程:

➜  ~ docker exec <container ID> ps -ef
UID          PID    PPID  C STIME TTY          TIME CMD
root           1       0  0 03:16 ?        00:00:00 ping -c 20 localhost
root           7       0  0 03:17 ?        00:00:00 ps -ef

可见,实际运行的命令,是把 ENTRYPOINT 的内容和 CMD 的内容组合起来了

  • ENTRYPOINT["ping", "-c", "20"]
  • CMD["localhost"]

最终命令是: ping -c 20 localhost

如果我们想ping另外一台主机,只需在 docker run 时覆盖 CMD 的值:

docker run kaidemo5 127.0.0.1

在另一个命令行窗口,查看容器:

➜  ~ docker ps
CONTAINER ID   IMAGE            COMMAND                  CREATED         STATUS         PORTS                      NAMES
b438242b4e84   kaidemo5         "ping -c 20 127.0.0.1"   3 seconds ago   Up 2 seconds                              intelligent_torvalds

查看容器里的进程:

➜  ~ docker exec <container ID> ps -ef
UID          PID    PPID  C STIME TTY          TIME CMD
root           1       0  0 03:25 ?        00:00:00 ping -c 20 127.0.0.1
root           8       0  0 03:25 ?        00:00:00 ps -ef

可见,通过覆盖 CMD 的值,就可以改变运行的参数。

在上面的例子里, -c 20 是写死在 ENTRYPOINT 里的,如果想要用户可以配置ping的次数,则应放在 CMD 里,以便用户在 docker run 时覆盖。

创建 Dockerfile 文件如下:

FROM ubuntu:trusty
ENTRYPOINT ["ping"]
CMD ["-c", "20", "localhost"]

构建:

docker build -t kaidemo6 .

启动容器:

docker run kaidemo6

效果和 docker run kaidemo5 是一样的,运行的都是 ping -c 20 localhost

启动容器时替换 CMD 参数:

docker run kaidemo6 -c 30 127.0.0.1

在另一个命令行窗口查看容器:

➜  ~ docker ps                      
CONTAINER ID   IMAGE            COMMAND                  CREATED         STATUS         PORTS                      NAMES
7885692cb508   kaidemo6         "ping -c 30 127.0.0.1"   4 seconds ago   Up 3 seconds                              wizardly_shtern

查看容器里的进程:

➜  ~ docker exec <container ID> ps -ef
UID          PID    PPID  C STIME TTY          TIME CMD
root           1       0  0 03:38 ?        00:00:00 ping -c 30 127.0.0.1
root           7       0  0 03:38 ?        00:00:00 ps -ef

可见,实际运行的是 ping -c 30 127.0.0.1

另外要注意的是,二者一定都要使用“exec形式”。如果使用“shell形式”,则会转为 /bin/sh xxx ,二者组合后,会造成混乱。

CMD localhostCMD [“localhost”]
ENTRYPOINT ping -c 10/bin/sh -c ‘ping -c 10’ /bin/sh -c localhost/bin/sh -c ‘ping -c 3’ localhost
ENTRYPOINT [“ping”,“-c”,“10”]ping -c 10 /bin/sh -c localhostping -c 3 localhost

可见,只有 二者都是“exec形式”时,才能组合出期望的结果。

总结: ENTRYPOINTCMD 组合使用:把运行命令和不变的参数放到 ENTRYPOINT 里,把可变的参数放到 CMD 里,以便在 docker run 时替换。二者都要使用“exec形式”。可用 --entrypoint 覆盖命令,但一般不这么做,因为 ENTRYPOINT 代表的是容器用途,一般不会改变,可变的是运行参数。

参考

  • https://spacelift.io/blog/docker-entrypoint-vs-cmd (里面有些内容和我实际测试结果不同,可能是Docker版本不同?)
  • https://zhuanlan.zhihu.com/p/30555962 (里面有些内容和我实际测试结果不同,可能是Docker版本不同?)
  • https://docs.docker.com/engine/reference/builder
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/duke_ding2/article/details/135418179

【docker】cmd entrypoint 区别 终极解读!_绝世好阿狸的博客-爱代码爱编程_entrypoint什么意思

昨天用Dockerfile来启动mongodb的集群,启动参数--replSet死活没执行,最后就决定研究一哈cmd和entrypoint。但是上网看了一些资料个人觉得讲的不好,还是没有说出根本的东西,决定自己研究并且整理一哈。 首先上docker官网:https://docs.docker.com/engine/reference/builder/#c

Dockerfile中CMD和ENTRYPOINT命令详解-爱代码爱编程

原文:https://m.jb51.net/article/136264.htm Dockerfile中CMD和ENTRYPOINT命令详解 发布时间:2018-03-12 11:46:00 作者:sparkdev Dockerfile中的ENTRYPOINT指令和CMD指令都可以设置容器启动时要执行的命令,但用途是有略微不同的。下面这篇文章主要给大

docker——cmd和entrypoint-爱代码爱编程

目录 1.copy和add的区别 2.cmd和entrypoint的区别 exec模式与shell模式 3.exec模式和shell模式 小实验 exec模式 使用exec模式无法输出环境变量 shell模式 cmd和entrypoint的总结 1.copy和add的区别 ADD :支持将远程URL的资源加入到镜像的文件系统 C

一起学Docker:CMD与ENTRYPOINT的区别-爱代码爱编程

CMD CMD指令为启动的容器指定默认要运行的程序,程序运行结束,容器也就结束。CMD 指令指定的程序可被 docker run 命令行参数中指定要运行的程序所覆盖。 类似于 RUN 指令,用于运行程序,但二者运行的时间点不同:CMD 在docker run 时运行,RUN 是在 docker build时运行。 注意:如果 Dockerfile 中如果

dockerfile 中的 cmd 和 entrypoint 有什么区别?-爱代码爱编程

问: 在 Dockerfiles 中有两个和我类似的命令:CMD 和 ENTRYPOINT。但我猜想它们之间存在(细微的?)差异——否则对于同一件事有两个命令是没有任何意义的。 CMD 的文档说明 CMD 的主要目的是

docker学习(二十二)dockerfile 中 cmd 与 entrypoint 区别-爱代码爱编程

目录 一、CMD 指令1.1 常见用法:1.2 用法1举例:1.3 docker run 覆盖 二、ENTRYPOINT 指令2.1 常见用法:2.2 两种写法: 三、ENTRYPOINT 指令和 CMD

docker 系列之 dockerfile 文件里 cmd命令与entrypoint命令区别-爱代码爱编程

文章目录 一、cmd:用法1:带有中括号的形式用法2:shell form,即没有中括号的形式 二、entrypoint:第一种:命令行模式,也就是带中括号第二种:shell模式 三、总结:

dockerfile(5) -爱代码爱编程

CMD 指定容器默认执行的命令 # exec 形式,推荐 CMD ["executable","param1","param2"] CMD ["可执行命令", "参数1", "参数2"...] # 作为ENTRYPOINT的默认参数 CMD ["param1","param2"] # shell CMD 命令 param1 param2 重

dockerfile中 cmd和entrypoint的区别-爱代码爱编程

在 Dockerfile 中,CMD 和 ENTRYPOINT 都用于指定容器启动时要执行的命令。它们之间的主要区别是: - CMD 用于定义容器启动时要执行的命令和参数,它设置的值可以被 Dockerfile 中的后续指令覆盖,包括在运行容器时传递的参数。如果在 Dockerfile 中没有指定 CMD,那么 Docker 将使用容器启动时所提供的默认

dockerfile中cmd命令的用法,你了解几个?_dockerfile cmd-爱代码爱编程

文章目录 1.问题描述2.解决方法3.问题原因分析4.CMD常见的用法4.1 执行应用程序4.2 使用 Shell 执行命令4.3 定义环境变量 5.最佳实践 1.问题描述 今天在

dockerfile中的cmd和entrypoint_dockerfile entrypoint和cmd-爱代码爱编程

在 Dockerfile 中,CMD 和 ENTRYPOINT 命令都用于指定容器启动时要执行的命令或可执行文件,但它们之间存在一些重要的区别。 CMD 命令 CMD 命令用于为容器指定默认的命令和参数。在一个 Dockerfile 中,通常只会有一个 CMD 命令,它可以有多个参数,这些参数通常是要执行的命令及其参数。当你使用 docker run