将你的 Go 镜像作为容器运行

先决条件

按照 构建你的 Go 镜像 中的步骤将 Go 应用容器化。

概述

在上一个模块中,你为示例应用创建了一个 Dockerfile,然后使用命令 docker build 创建了 Docker 镜像。现在你有了镜像,可以运行该镜像并查看你的应用是否正常运行。

容器是一个普通的操作系统进程,只不过这个进程是隔离的,拥有自己的文件系统、自己的网络和独立于主机的独立进程树。

要在容器内运行镜像,请使用 docker run 命令。它需要一个参数,即镜像名称。启动你的镜像并确保其正常运行。在你的终端中运行以下命令。

$ docker run docker-gs-ping
   ____    __
  / __/___/ /  ___
 / _// __/ _ \/ _ \
/___/\__/_//_/\___/ v4.10.2
High performance, minimalist Go web framework
https://echo.labstack.com
____________________________________O/_______
                                    O\
⇨ http server started on [::]:8080

运行此命令时,你会注意到并没有返回到命令行提示符。这是因为你的应用是一个 REST 服务器,它将在循环中等待传入请求,直到你停止容器才会将控制权返回给操作系统。

使用 curl 命令向服务器发起 GET 请求。

$ curl http://localhost:8080/
curl: (7) Failed to connect to localhost port 8080: Connection refused

你的 curl 命令失败了,因为连接到你的服务器被拒绝。这意味着你无法连接到 localhost 的 8080 端口。这是预期的,因为你的容器是隔离运行的,其中包括网络。停止容器并在本地网络上发布 8080 端口后重新启动。

要停止容器,请按 ctrl-c。这将返回到终端提示符。

要为你的容器发布端口,请在 docker run 命令上使用 --publish 标志(简写为 -p)。--publish 命令的格式为 [主机端口]:[容器端口]。因此,如果你想将容器内的 8080 端口暴露给容器外的 3000 端口,你需要向 --publish 标志传递 3000:8080

启动容器并将容器内的 8080 端口暴露给主机的 8080 端口。

$ docker run --publish 8080:8080 docker-gs-ping

现在,重新运行 curl 命令。

$ curl http://localhost:8080/
Hello, Docker! <3

成功!你已连接到在容器内 8080 端口运行的应用。切换回运行容器的终端,你应该会看到 GET 请求被记录到控制台。

ctrl-c 停止容器。

在分离模式下运行

到目前为止一切顺利,但你的示例应用是一个 Web 服务器,你不应该需要将终端连接到容器。Docker 可以在后台以分离模式运行你的容器。为此,你可以使用 --detach 或其简写 -d 标志。Docker 将像之前一样启动你的容器,但这次会与容器分离,并将控制权返回给你终端提示符。

$ docker run -d -p 8080:8080 docker-gs-ping
d75e61fcad1e0c0eca69a3f767be6ba28a66625ce4dc42201a8a323e8313c14e

Docker 在后台启动了你的容器,并在终端上打印了容器 ID。

再次确认你的容器正在运行。运行相同的 curl 命令。

$ curl http://localhost:8080/
Hello, Docker! <3

列出容器

既然你在后台运行了容器,你怎么知道你的容器是否正在运行,或者你的机器上还有哪些其他容器正在运行?要查看你的机器上正在运行的容器列表,请运行 docker ps 命令。这类似于在 Linux 机器上使用 ps 命令查看进程列表的方式。

$ docker ps

CONTAINER ID   IMAGE            COMMAND             CREATED          STATUS          PORTS                    NAMES
d75e61fcad1e   docker-gs-ping   "/docker-gs-ping"   41 seconds ago   Up 40 seconds   0.0.0.0:8080->8080/tcp   inspiring_ishizaka

ps 命令会告诉你关于正在运行的容器的许多信息。你可以看到容器 ID、在容器内运行的镜像、用于启动容器的命令、创建时间、状态、暴露的端口以及容器的名称。

你可能想知道你的容器名称是从哪里来的。由于你在启动容器时没有指定名称,Docker 生成了一个随机名称。稍后我们会解决这个问题,但首先你需要停止容器。要停止容器,请运行 docker stop 命令,并传入容器的名称或 ID。

$ docker stop inspiring_ishizaka
inspiring_ishizaka

现在重新运行 docker ps 命令以查看正在运行的容器列表。

$ docker ps

CONTAINER ID   IMAGE     COMMAND   CREATED   STATUS    PORTS     NAMES

停止、启动和命名容器

Docker 容器可以被启动、停止和重启。当你停止一个容器时,它不会被移除,但状态会变为停止,容器内的进程也会停止。当你运行 docker ps 命令时,默认输出只显示正在运行的容器。如果你传递 --all 标志(简写为 -a),你将看到系统上的所有容器,包括已停止的容器和正在运行的容器。

$ docker ps --all

CONTAINER ID   IMAGE            COMMAND                  CREATED              STATUS                      PORTS     NAMES
d75e61fcad1e   docker-gs-ping   "/docker-gs-ping"        About a minute ago   Exited (2) 23 seconds ago             inspiring_ishizaka
f65dbbb9a548   docker-gs-ping   "/docker-gs-ping"        3 minutes ago        Exited (2) 2 minutes ago              wizardly_joliot
aade1bf3d330   docker-gs-ping   "/docker-gs-ping"        3 minutes ago        Exited (2) 3 minutes ago              magical_carson
52d5ce3c15f0   docker-gs-ping   "/docker-gs-ping"        9 minutes ago        Exited (2) 3 minutes ago              gifted_mestorf

如果你一直跟着操作,你应该会看到列出了几个容器。这些是你已经启动和停止但尚未移除的容器。

重启刚刚停止的容器。找到容器的名称,并将以下 restart 命令中的容器名称替换为你系统上的容器名称。

$ docker restart inspiring_ishizaka

现在,再次使用 ps 命令列出所有容器。

$ docker ps -a

CONTAINER ID   IMAGE            COMMAND                  CREATED          STATUS                     PORTS                    NAMES
d75e61fcad1e   docker-gs-ping   "/docker-gs-ping"        2 minutes ago    Up 5 seconds               0.0.0.0:8080->8080/tcp   inspiring_ishizaka
f65dbbb9a548   docker-gs-ping   "/docker-gs-ping"        4 minutes ago    Exited (2) 2 minutes ago                            wizardly_joliot
aade1bf3d330   docker-gs-ping   "/docker-gs-ping"        4 minutes ago    Exited (2) 4 minutes ago                            magical_carson
52d5ce3c15f0   docker-gs-ping   "/docker-gs-ping"        10 minutes ago   Exited (2) 4 minutes ago                            gifted_mestorf

注意到你刚刚重启的容器已在分离模式下启动,并暴露了 8080 端口。另外,注意容器的状态是 Up X seconds。当你重启容器时,它将使用最初启动时所用的相同标志或命令来启动。

停止并移除所有容器,然后看看如何解决随机命名问题。

停止你刚刚启动的容器。找到正在运行的容器的名称,并将以下命令中的名称替换为你系统上的容器名称。

$ docker stop inspiring_ishizaka
inspiring_ishizaka

现在所有容器都已停止,移除它们。当容器被移除时,它不再运行,也不处于停止状态。相反,容器内的进程被终止,容器的元数据被移除。

要移除容器,请运行 docker rm 命令并传入容器名称。你可以在一个命令中传入多个容器名称。

再次确保将以下命令中的容器名称替换为你系统上的容器名称。

$ docker rm inspiring_ishizaka wizardly_joliot magical_carson gifted_mestorf

inspiring_ishizaka
wizardly_joliot
magical_carson
gifted_mestorf

再次运行 docker ps --all 命令,验证所有容器是否已移除。

现在,解决恼人的随机命名问题。标准做法是为你的容器命名,原因很简单:更容易识别容器中运行的是什么,以及它与哪个应用或服务相关联。就像代码中良好的变量命名规范使其更易于阅读一样,为容器命名也是如此。

要为容器命名,你必须向 run 命令传递 --name 标志。

$ docker run -d -p 8080:8080 --name rest-server docker-gs-ping
3bbc6a3102ea368c8b966e1878a5ea9b1fc61187afaac1276c41db22e4b7f48f
$ docker ps

CONTAINER ID   IMAGE            COMMAND             CREATED          STATUS          PORTS                    NAMES
3bbc6a3102ea   docker-gs-ping   "/docker-gs-ping"   25 seconds ago   Up 24 seconds   0.0.0.0:8080->8080/tcp   rest-server

现在,你可以根据名称轻松识别你的容器。

下一步

在本模块中,你学习了如何运行容器和发布端口。你还学习了如何管理容器的生命周期。然后,你学习了为容器命名的重要性,以便它们更容易被识别。在下一个模块中,你将学习如何在容器中运行数据库并将其连接到你的应用。

页面选项