云上的天涯

导航

Dockerfile的CMD总结

最近又在写Dockerfile,在写的过程中对CMD又测试了很多,对应的还有个ENTRYPOINT,也在网上找了很多资料,大概总结如下

先来大概说下Dockerfile中可以执行命令的指令,以下非原创,有出处

Dockerfile中RUN,CMD和ENTRYPOINT都能够用于执行命令,下面是三者的主要用途:

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

Shell格式和Exec格式运行命令

我们可用两种方式指定 RUN、CMD 和 ENTRYPOINT 要运行的命令:Shell 格式和 Exec 格式:

  • Shell格式:<instruction> <command>。例如:apt-get install python3
  • Exec格式:<instruction> ["executable", "param1", "param2", ...]。例如: ["apt-get", "install", "python3"]

CMD 和 ENTRYPOINT 推荐使用 Exec 格式,因为指令可读性更强,更容易理解。RUN 则两种格式都可以。

作者:伊凡的一天
链接:https://www.jianshu.com/p/f0a0f6a43907
来源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。
View Code
docker CMD 有三种形式
CMD ["executable","param1","param2"] 使用 exec 执行,推荐方式;
CMD command param1 param2 在 /bin/sh 中执行,提供给需要交互的应用;
CMD ["param1","param2"] 提供给 ENTRYPOINT 的默认参数;
指定启动容器时执行的命令,每个 Dockerfile 只能有一条 CMD 命令。如果指定了多条命令,只有最后一条会被执行。
如果用户启动容器时候指定了运行的命令,则会覆盖掉 CMD 指定的命令。
shell 格式的话如下
CMD  echo "hello"
exec 格式的话如下
CMD ["echo","hello"]
exec 格式必须用双引号,中间用逗号隔开,执行命令 和命令后参数都是用双引号引起来,每个都要用逗号隔开。
shell 格式和 exec 格式主要区别在于

shell 格式的话,实际的命令会被包装为 sh -c 的参数的形式进行执行。比如:

CMD echo $HOME

在实际执行中,会将其变更为:

CMD [ "sh", "-c", "echo $HOME" ]

这就是为什么我们可以使用环境变量的原因,因为这些环境变量会被 shell 进行解析处理,CMD会使用容器内的shell变量,自定义ENV
如果使用exec 格式的话,CMD 不会使用容器内的变量和自定义ENV
比如我们自定义了ENV name=lisa
使用shell模式的话,可以输出$name 为lisa
但是exec 模式的话是不能正常输出$namd的
如果想用exec 模式获取ENV变量的话,可以用折中方法,就是编写脚本,然后使用脚本去获取变量输出,CMD指令执行脚本,脚本一定加可执行权限比如 CMD["/root/test.sh"]
test.sh内容为
#!/bin/bash
echo $name

chmo 755 test.sh

这样打包进镜像后,就可以正常输出变量了,因为脚本使用的也是容器的shell,会被shell进行解析处理

最后一种运行模式
CMD ["参数1","参数2"] 是配合 ENTRYPOINT 使用的
CMD 和 ENTRYPOINT 最大区别在于,CMD 指定在容器启动时候指定了指令就会覆盖CMD
但是 ENTRYPOINT 不会被覆盖,只能追加参数,CMD可以配合ENTRYPOINT 使用,可以给ENTRYPOINT追加参数,另外两者结合使用的时候 CMD 指令是可以被覆盖的。

posted on 2019-06-27 14:37  云上的天涯  阅读(1645)  评论(0编辑  收藏  举报