- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章Dockerfile中CMD和ENTRYPOINT命令详解由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
前言 。
cmd 和 entrypoint 指令都是用来指定容器启动时运行的命令.
单从功能上来看,这两个命令几乎是重复的。单独使用其中的一个就可以实现绝大多数的用例。但是既然 doker 同时提供了它们,为了在使用中不至于混淆,本文试图把它们的用法理清楚。下面话不多说了,来一起看看详细的介绍吧.
exec 模式和 shell 模式 。
cmd 和 entrypoint 指令都支持 exec 模式和 shell 模式的写法,所以要理解 cmd 和 entrypoint 指令的用法,就得先区分 exec 模式和 shell 模式。这两种模式主要用来指定容器中的不同进程为 1 号进程。了解 linux 的朋友应该清楚 1 号进程在系统中的重要地位。笔者也在《在 docker 容器中捕获信号》一文中介绍过 1 号进程对容器中信号处理的重要性,感兴趣的朋友可以移步这里进行了解。下面我们通过 cmd 指令来学习 exec 模式和 shell 模式的特点.
exec 模式 。
使用 exec 模式时,容器中的任务进程就是容器内的 1 号进程,看下面的例子:
1
2
|
from ubuntu
cmd [
"top"
]
|
把上面的代码保存到 test1 目录的 dockerfile 中,然后进入 test1 目录构建镜像并启动一个容器:
1
2
|
$ docker build -t test1 .
$ docker run -idt --name testcon test1
|
然后查看容器中的进程 id:
1
|
$ docker
exec
testcon
ps
aux
|
从图中我们看到运行 top 命令的进程 id 为 1.
exec 模式是建议的使用模式,因为当运行任务的进程作为容器中的 1 号进程时,我们可以通过 docker 的 stop 命令优雅的结束容器(详情请参考《在 docker 容器中捕获信号》).
exec 模式的特点是不会通过 shell 执行相关的命令,所以像 $home 这样的环境变量是取不到的:
1
2
|
from ubuntu
cmd [
"echo"
,
"$home"
]
|
把上面的代码保存到 test1 目录的 dockerfile 中,然后进入 test1 目录构建镜像并启动一个容器:
1
2
|
$ docker build --no-cache -t test1 .
$ docker run --
rm
test1
|
通过 exec 模式执行 shell 可以获得环境变量:
1
2
|
from ubuntu
cmd [
"sh"
,
"-c"
,
"echo $home"
]
|
把上面的代码保存到 test1 目录的 dockerfile 中,然后进入 test1 目录构建镜像并启动一个容器:
1
2
|
$ docker build --no-cache -t test1 .
$ docker run --
rm
test1
|
这次正确取到了 $home 环境变量的值.
shell 模式 。
使用 shell 模式时,docker 会以 /bin/sh -c "task command" 的方式执行任务命令。也就是说容器中的 1 号进程不是任务进程而是 bash 进程,看下面的例子:
1
2
|
from ubuntu
cmd
top
|
把上面的代码保存到 test2 目录的 dockerfile 中,然后进入 test2 目录构建镜像并启动一个容器:
1
2
|
$ docker build -t test2 .
$ docker run -itd --name testcon2 test2
|
然后查看容器中的进程 id:
1
|
$ docker
exec
testcon2
ps
aux
|
1 号进程执行的命令居然是 /bin/sh -c top。而我们指定的 top 命令的进程 id 为 7。这是由 docker 内部决定的,目的是让我们执行的命令或者脚本可以取到环境变量.
cmd 指令 。
cmd 指令的目的是:为容器提供默认的执行命令.
cmd 指令有三种使用方式,其中的一种是为 entrypoint 提供默认的参数:
1
|
cmd [
"param1"
,
"param2"
]
|
另外两种使用方式分别是 exec 模式和 shell 模式:
1
2
|
cmd [
"executable"
,
"param1"
,
"param2"
]
//
这是
exec
模式的写法,注意需要使用双引号。
cmd
command
param1 param2
//
这是 shell 模式的写法。
|
注意命令行参数可以覆盖 cmd 指令的设置,但是只能是重写,却不能给 cmd 中的命令通过命令行传递参数。 一般的镜像都会提供容器启动时的默认命令,但是有些场景中用户并不想执行默认的命令。用户可以通过命令行参数的方式覆盖 cmd 指令提供的默认命令。比如通过下面命令创建的镜像:
1
2
|
from ubuntu
cmd [
"top"
]
|
在启动容器时我们通过命令行指定参数 ps aux 覆盖默认的 top 命令:
从上图可以看到,命令行上指定的 ps aux 命令覆盖了 dockerfile 中的 cmd [ "top" ]。实际上,命令行上的命令同样会覆盖 shell 模式的 cmd 指令.
entrypoint 指令 。
entrypoint 指令的目的也是为容器指定默认执行的任务.
entrypoint 指令有两种使用方式,就是我们前面介绍的 exec 模式和 shell 模式:
1
2
|
entrypoint [
"executable"
,
"param1"
,
"param2"
]
//
这是
exec
模式的写法,注意需要使用双引号。
entrypoint
command
param1 param2
//
这是 shell 模式的写法。
|
exec 模式和 shell 模式的基本用法和 cmd 指令是一样的,下面我们介绍一些比较特殊的用法.
指定 entrypoint 指令为 exec 模式时,命令行上指定的参数会作为参数添加到 entrypoint 指定命令的参数列表中。用下面的代码构建镜像 test1:
1
2
|
from ubuntu
entrypoint [
"top"
,
"-b"
]
|
运行下面的命令:
1
|
$ docker run --
rm
test1 -c
|
我们在命令行上添加的参数被追加到了 top 命令的参数列表中.
由 cmd 指令指定默认的可选参数:
1
2
3
|
from ubuntu
entrypoint [
"top"
,
"-b"
]
cmd [
"-c"
]
|
使用这段代码构建镜像 test2 并不带命令行参数启动容器:
1
|
$ docker run --
rm
test2
|
这时容器中运行的命令为:top -b -c.
如果我们指定命令行参数:
1
|
$ docker run --
rm
test2 -n 1
|
-n 1 会覆盖 通过 cmd [ "-c" ] 指定的参数,容器执行的命令为:top -b -n 1 。
注意上图的输出显示 -c 参数被覆盖了.
指定 entrypoint 指令为 shell 模式时,会完全忽略命令行参数:
1
2
|
from ubuntu
entrypoint
echo
$home
|
把上面的代码编译成镜像 test2,分别不带命令行参数和使用命令行参数 ls 执行命令:
我们看到 ls 命令没有被执行,这说明命令行参数被 entrypoint 指令的 shell 模式忽略了.
覆盖默认的 entrypoint 指令:
entrypoint 指令也是可以被命令行覆盖的,只不过不是默认的命令行参数,而是需要显式的指定 --entrypoint 参数。比如我们通过下面的方式覆盖上面镜像中的 echo $home 命令:
1
|
$ docker run --
rm
--entrypoint
hostname
test2
|
这里我们使用 hostname 命令覆盖了默认的 echo $home 命令.
dockerfile 中至少要有一个 。
如果镜像中既没有指定 cmd 也没有指定 entrypoint 那么在启动容器时会报错。这不算是什么问题,因为现在能见到的绝大多数镜像都默认添加了 cmd 或 entrypoint 指令.
指定任意一个,效果差不多 。
从结果上看,cmd 和 entrypoint 是一样的,我们可以通过它们实现相同的目的。下面我们分别用 cmd 和 entrypoint 设置 top -b 命令,然后观察容器运行时的 metadata 信息:
或者:
虽然实现方式不同,但最终容器运行的命令是一样的.
同时使用 cmd 和 entrypoint 的情况 。
对于 cmd 和 entrypoint 的设计而言,多数情况下它们应该是单独使用的。当然,有一个例外是 cmd 为 entrypoint 提供默认的可选参数.
我们大概可以总结出下面几条规律:
• 如果 entrypoint 使用了 shell 模式,cmd 指令会被忽略.
• 如果 entrypoint 使用了 exec 模式,cmd 指定的内容被追加为 entrypoint 指定命令的参数.
• 如果 entrypoint 使用了 exec 模式,cmd 也应该使用 exec 模式.
真实的情况要远比这三条规律复杂,好在 docker 给出了官方的解释,如下图所示:
当我们无法理解容器中运行命令的行为时,说不定通过这个表格可以解开疑惑! 。
总结 。
对于 dockerfile 来说,cmd 和 entrypoint 是非常重要的指令。它们不是在构建镜像的过程中执行,而是在启动容器时执行,所以主要用来指定容器默认执行的命令。但是提供两个功能类似的指令,必然会给用户带来理解上的困惑和使用中的混淆。希望本文能够帮助大家理解二者的区别与联系,并更好的使用二者.
参考:
docker 官方文档 entrypoint vs cmd: back to basics dockerfile: entrypoint vs cmd 。
原文链接:http://www.cnblogs.com/sparkdev/p/8461576.html 。
最后此篇关于Dockerfile中CMD和ENTRYPOINT命令详解的文章就讲到这里了,如果你想了解更多关于Dockerfile中CMD和ENTRYPOINT命令详解的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我正在尝试使用谷歌云构建触发器来构建我的 docker 容器。不幸的是,构建没有找到我的 docker 文件。它不在我的存储库的根目录中,但我认为我可以指定一个相对路径。显然我配置错误了。下面是我的
我可以像这样构建一个 Dockerfile docker build -t deepak/ruby . 但是对于一个不名为Dockerfile的Dockerfile # DOCKER-VERSION
是否可以命名构建阶段?我正在搜索类似以下示例的内容: ARG NAME FROM python:3.7-alpine as modul-${NAME} # ... 如果我尝试这个例子,就会发生这个错误
在我的 Dockerfile 中,我需要使用命令替换来添加一些环境变量。我想设置 ENV PYTHONPATH /usr/local/$(python3 -c 'from distutils impo
以下Dockerfile包含四个COPY层: COPY README.md ./ COPY package.json ./ COPY gulpfile.js ./ COPY __BUILD_NUMBE
我正在尝试从我创建的 Dockerfile 中拉取 tomcat,并使用以下命令拉取 centos: FROM centos RUN docker run -it tomcat 然后现在我正在创建一个
我不确定是否应该为我的 Node.js 应用程序创建不同的 Dockerfile 文件。一种用于没有开发依赖项的生产,另一种用于包含开发依赖项的测试。 或者一个基本上是开发Dockerfile.dev
我拥有的是多个相似且简单的dockerfile 但我想要的是有一个单一的基础 dockerfile 和我的 dockerfile将他们的变量传递给它。 在我的例子中,dockerfile 之间的唯一区
我有一个应用程序有两个 Dockerfile ,说 Dockerfile.foo和 Dockerfile.bar ,预计将为 docker-compose.yml 的两个不同服务生成两个不同的图像.
我是一个 docker maven 插件新手。 如果我理解得很好,根据Spotify's Dockerfile Maven documentation ,应该将 Dockerfile 放在我的项目的根
我正在设置一个docker容器,将在this article之后在nginx服务器上为我的Angular 5应用程序提供服务。 本文提出了这个Dockerfile: # Stage 0, based
大家好, 我是 docker 的新手,我想容器化我的 ASP.NET Core 应用程序,但出现错误,我无法继续构建我的图像 这是我的示例 Dockerfile FROM microsoft/dotn
我想在不使用注册表的情况下链接 Dockerfiles,这可以通过 FROM 实现吗?陈述? 以两个 Dockerfile 为例: Dockerfile /some/other/dir/Dockerf
我尝试在公共(public) dockerhub 上通过我自己的镜像安装 Airflow,但它在本地运行完美,但当我尝试在 Openshift 上使用它时。我在下面收到此错误。 `ERROR: Cou
此 dockerfile 工作正常。但是如何在 dockerfile 中执行命令? FROM alpine RUN apk add --update sqlite && rm -rf /var/cac
当我尝试设置自动构建时,我的 Docker Hub 存储库中的 Dockerfile 为空。如果我触发构建:- Build failed: Dockerfile not found at ./Dock
关闭。这个问题需要details or clarity .它目前不接受答案。 想改进这个问题?通过 editing this post 添加详细信息并澄清问题. 1年前关闭。 Improve this
我目前正在将一个巨大的 DockerFile 拆分为单独的 DockerFile,但是在巨大的 DockerFile 中定义的一些变量需要在单独的 DockerFile 中使用。 Docker 是否有
我有一个 Dockerfile 可以创建我想在这里使用的构建镜像:~/build/Dockerfile 然后我使用一个标准镜像来部署 从 ~/build/Dockerfile 构建的图像没有在任何地方
我运行这个: ➜ frontend git:(master) ✗ docker-compose up Building web ERROR: Cannot locate specified Dock
我是一名优秀的程序员,十分优秀!