Dckerfile CMD不同格式的区别

原文:https://www.cnblogs.com/hh2737/p/11096872.html

参考:https://www.cnblogs.com/lnlvinso/p/13308585.html

最近发现一个docker镜像的Dockefile中没有CMD,请教了下别人,也学了些知识。

     

     Dockefile中没有CMD,那么会去调用父镜像的CMD。

     这样做还有一个好处是,各个不同子镜像可以对同一个环境变量设置不同的值,这样父镜像通过shell取到设置的环境变量,同一个CMD实际运行的是子镜像的变量。

 

      原文:https://www.cnblogs.com/hh2737/p/11096872.html

  最近又在写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 则两种格式都可以。

  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 @ 2022-04-06 22:00  你压我头发了~  阅读(76)  评论(0编辑  收藏  举报