根据我目前阅读的教程,使用"docker run -d"将从image启动一个容器,并且容器将在后台运行。这就是它的样子,我们已经有了container id。

root@docker:/home/root# docker run -d centos
605e3928cdddb844526bab691af51d0c9262e0a1fc3d41de3f59be1a58e1bd1d

但如果我运行“docker ps”,什么都没有返回。

所以我尝试了“docker ps -a”,我可以看到容器已经退出:

root@docker:/home/root# docker ps -a
CONTAINER ID        IMAGE                 COMMAND             CREATED             STATUS                         PORTS               NAMES
605e3928cddd        centos:latest         "/bin/bash"         31 minutes ago      Exited (0) 31 minutes ago                          kickass_swartz

我做错什么了吗?如何解决此问题?


当前回答

背景

Docker容器运行一个进程(“命令”或“入口点”)来保持它的活性。只要命令继续运行,容器就会继续运行。

在您的例子中,命令(默认情况下,centos:latest上的/bin/bash)立即退出(bash在没有连接到终端并且没有什么要运行时也会退出)。

通常,当您以守护进程模式(使用-d)运行容器时,容器正在运行某种守护进程(如httpd)。在这种情况下,只要httpd守护进程在运行,容器就会保持活动状态。

您似乎要做的是在容器内不运行守护进程的情况下保持容器的活性。这有点奇怪(因为容器在与它交互之前没有做任何有用的事情,可能是与docker exec交互),但在某些情况下,这样做可能是有意义的。

(您的意思是在容器内获得bash提示符吗?这很简单!Docker运行- centos:最新)

解决方案

让容器无限期地处于守护模式下活动的一个简单方法是运行sleep infinity作为容器的命令。这并不依赖于做一些奇怪的事情,比如在守护进程模式下分配TTY。尽管它确实依赖于一些奇怪的事情,比如把睡眠作为你的主要命令。

$ docker run -d centos:latest sleep infinity
$ docker ps
CONTAINER ID  IMAGE         COMMAND          CREATED       STATUS       PORTS NAMES
d651c7a9e0ad  centos:latest "sleep infinity" 2 seconds ago Up 2 seconds       nervous_visvesvaraya

可选择的解决方案

正如cjsimon所指出的,-t选项分配一个“伪tty”。这个技巧会继续无限地运行,因为它认为它连接到一个交互式TTY(即使如果不传递-i,您没有办法与那个特定的TTY进行交互)。不管怎样,这个也可以做到:

$ docker run -t -d centos:latest

不能100%确定-t是否会产生其他奇怪的相互作用;如果是的话,请在下方留言。

其他回答

背景

Docker容器运行一个进程(“命令”或“入口点”)来保持它的活性。只要命令继续运行,容器就会继续运行。

在您的例子中,命令(默认情况下,centos:latest上的/bin/bash)立即退出(bash在没有连接到终端并且没有什么要运行时也会退出)。

通常,当您以守护进程模式(使用-d)运行容器时,容器正在运行某种守护进程(如httpd)。在这种情况下,只要httpd守护进程在运行,容器就会保持活动状态。

您似乎要做的是在容器内不运行守护进程的情况下保持容器的活性。这有点奇怪(因为容器在与它交互之前没有做任何有用的事情,可能是与docker exec交互),但在某些情况下,这样做可能是有意义的。

(您的意思是在容器内获得bash提示符吗?这很简单!Docker运行- centos:最新)

解决方案

让容器无限期地处于守护模式下活动的一个简单方法是运行sleep infinity作为容器的命令。这并不依赖于做一些奇怪的事情,比如在守护进程模式下分配TTY。尽管它确实依赖于一些奇怪的事情,比如把睡眠作为你的主要命令。

$ docker run -d centos:latest sleep infinity
$ docker ps
CONTAINER ID  IMAGE         COMMAND          CREATED       STATUS       PORTS NAMES
d651c7a9e0ad  centos:latest "sleep infinity" 2 seconds ago Up 2 seconds       nervous_visvesvaraya

可选择的解决方案

正如cjsimon所指出的,-t选项分配一个“伪tty”。这个技巧会继续无限地运行,因为它认为它连接到一个交互式TTY(即使如果不传递-i,您没有办法与那个特定的TTY进行交互)。不管怎样,这个也可以做到:

$ docker run -t -d centos:latest

不能100%确定-t是否会产生其他奇怪的相互作用;如果是的话,请在下方留言。

我也遇到了同样的问题,只是打开另一个终端,对我来说很管用:

创建容器:

docker run -d mcr.microsoft.com/mssql/server:2019-CTP3.0-ubuntu
containerid=52bbc9b30557

启动容器:

docker start 52bbc9b30557

启动bash以保持容器运行:

docker exec -it 52bbc9b30557 bash

启动流程您需要:

docker exec -it 52bbc9b30557 /path_to_cool_your_app

Docker需要你的命令保持在前台运行。否则,它会认为应用程序停止并关闭容器。

因此,如果你的docker输入脚本是一个后台进程,如下所示:

/usr/local/bin/confd -interval=30 -backend etcd -node $CONFIG_CENTER &

如果以后没有触发其他前台进程,'&'将使容器停止并退出。 因此,解决方案是删除'&'或在它后面运行另一个前台CMD,例如

tail -f server.log

你可以通过以下任何一种方式实现你想要的:

docker run -t -d <image-name>

or

docker run -i -d <image-name>

or

docker run -it -d <image-name>

其他答案所建议的命令参数(即tail -f /dev/null)是完全可选的,并不是让容器保持在后台运行所必需的。

还要注意Docker文档建议组合-i和-t选项会导致它像shell一样运行。

See:

https://docs.docker.com/engine/reference/run/#foreground

论证顺序很重要

Jersey Beans的答案(所有3个例子)对我很有用。经过相当多的尝试和错误之后,我意识到论点的顺序很重要。

保持容器在后台运行: Docker运行-t -d <image-name>

保持容器在前台运行:docker run <image-name> -t -d

对于我这种拥有Powershell背景的人来说,这一点并不明显。