容器概念

容器是依赖于镜像生成,独立运行的一个或一组应用以及它们的运行环境。

启动容器

启动容器有两种方式

  • 基于镜像新建一个容器并启动
  • 将在终止状态(stopped)的容器重新启动

    新建并启动 (docker run)

    比如下面的命令输出“hello world!”,之后终止容器
    1. [root@wangpengliang ~]# docker run ubuntu /bin/echo 'hello world!'
    2. hello world!
    下面的命令则启动一个 bash 终端,允许用户进行交互
    1. [root@wangpengliang ~]# docker run -it ubuntu /bin/bash
    2. root@ef8fa7a1ffd6:/#
    -t 选项让Docker分配一个伪终端(pseudo-tty)并绑定到容器的标准输入上, -i 则让容器的标准输入保持打开。 -it 是简写
    在交互模式下可以通过所创建的终端来输入命令,比如:
    1. [root@wangpengliang ~]# docker run -it ubuntu /bin/bash
    2. root@ef8fa7a1ffd6:/# pwd
    3. /
    4. root@ef8fa7a1ffd6:/# ls
    5. bin boot dev etc home lib lib32 lib64 libx32 media mnt opt proc root run sbin srv sys tmp usr var
    6. root@ef8fa7a1ffd6:/#
    当利用 docker run 创建容器时,Docker 在后台运行的标准操作包括:
  1. 检查本地是否存在指定的镜像,不存在就从公有仓库下载
  2. 利用镜像创建并启动一个容器
  3. 分配一个文件系统,并在只读的镜像层外面挂载一层可读写层
  4. 从宿主主机配置的网桥接口中桥接一个虚拟接口到容器中去
  5. 从地址池配置一个 ip 地址给容器
  6. 执行用户指定的应用程序
  7. 执行完毕后容器被终止

    启动已终止容器

    可以利用 docker container start 命令,直接将一个已经终止的容器启动运行。容器的核心为所执行的应用程序,所需要的资源都是应用程序运行所必需的。除此之外,并没有其它的资源。可以在伪终端中利用 pstop 来查看进程信息
    1. root@ef8fa7a1ffd6:/# ps
    2. PID TTY TIME CMD
    3. 1 pts/0 00:00:00 bash
    4. 12 pts/0 00:00:00 ps

    后台运行容器

    更多的时候,需要让 Docker 在后台运行而不是直接把执行命令的结果输出在当前宿主机下。可以通过添加 -d 参数来实现。

    不使用 -d 参数运行

    1. [root@wangpengliang ~]# docker run ubuntu /bin/sh -c 'while true;do echo hello docker; sleep 1;done'
    2. hello docker
    3. hello docker
    4. hello docker
    5. hello docker
    容器会把输出的结果 (STDOUT) 打印到宿主机上1s打印一次”hello docker”。

    使用了 -d 参数运行

    1. [root@wangpengliang ~]# docker run -d ubuntu /bin/sh -c 'while true;do echo hello docker; sleep 1;done'
    2. 3c443074634c5edd6272c62fd08d8439ac2b5f3b9677233aae06deb2724c917c
    此时容器会在后台运行并不会把输出结果 (STDOUT) 打印到宿主机上,可以使用 docker logs 查看输出结果
    1. [root@wangpengliang ~]# docker logs 3c443074634c
    2. hello docker
    3. hello docker
    4. hello docker

    注意: 容器是否会长久运行,和 docker run 指定的命令有关,和 -d 参数无关

使用 -d 参数启动后会返回一个唯一的 id,也可以通过 docker container ls 命令来查看容器信息

  1. [root@wangpengliang ~]# docker container ls
  2. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
  3. 3c443074634c ubuntu "/bin/sh -c 'while t…" 2 minutes ago Up 2 minutes affectionate_agnesi
  4. a1ea64a6c039 nginx:3.0 "/docker-entrypoint.…" 17 hours ago Up 17 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3

要获取容器的输出信息,可以通过 docker container logs 命令,格式 $ docker container logs [container ID or NAMES]

  1. [root@wangpengliang ~]# docker container logs 3c443074634c
  2. hello docker
  3. hello docker
  4. hello docker

终止容器

可以使用 docker container stop 来终止一个运行中的容器,此外当 Docker 容器中指定的应用终结时,容器也自动终止。

比如对于上面只启动了一个终端的容器,通过 exit 命令或 ctrl+d 退出终端时,创建的容器立刻终止。终止状态的容器可以用 docker container ls -a 命令看到

  1. [root@wangpengliang ~]# docker container ls -a
  2. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
  3. a1ea64a6c039 nginx:3.0 "/docker-entrypoint.…" 18 hours ago Up 18 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3
  4. bab7ffb6e2e9 nginx:2.0 "/docker-entrypoint.…" 3 days ago Exited (255) 18 hours ago 0.0.0.0:81->80/tcp, :::81->80/tcp webserver2
  5. 470e01aed950 nginx

处于终止状态的容器,可以通过 docker container start 命令来重新启动

  1. [root@wangpengliang ~]# docker container start webserver
  2. webserver
  3. [root@wangpengliang ~]# docker ps -a
  4. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
  5. a1ea64a6c039 nginx:3.0 "/docker-entrypoint.…" 18 hours ago Up 18 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3
  6. bab7ffb6e2e9 nginx:2.0 "/docker-entrypoint.…" 3 days ago Exited (255) 18 hours ago 0.0.0.0:81->80/tcp, :::81->80/tcp webserver2
  7. 470e01aed950 nginx "/docker-entrypoint.…" 3 days ago Up 5 seconds 0.0.0.0:80->80/tcp, :::80->80/tcp webserver

此外 docker container restart 命令会将一个运行状态的容器终止,然后再重新启动。

进入容器

使用 -d 参数时,容器启动后会进入后台,某些时候需要进入容器进行操作,包括使用 docker attachdocker exec 命令,推荐使用 docker exec

attach

  1. [root@wangpengliang ~]# docker run -dit ubuntu
  2. 21498e13f12425718093684e74f342e30e7b9665e57b9b919f0a718d888f7b50
  3. [root@wangpengliang ~]# docker container ls
  4. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
  5. 21498e13f124 ubuntu "/bin/bash" 9 seconds ago Up 8 seconds elated_chatelet
  6. a1ea64a6c039 nginx:3.0 "/docker-entrypoint.…" 18 hours ago Up 18 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3
  7. 470e01aed950 nginx "/docker-entrypoint.…" 3 days ago Up 2 minutes 0.0.0.0:80->80/tcp, :::80->80/tcp webserver
  8. [root@wangpengliang ~]# docker attach 21498e13f124
  9. root@21498e13f124:/# ls
  10. bin boot dev etc home lib lib32 lib64 libx32 media mnt opt proc root run sbin srv sys tmp usr var
  11. root@21498e13f124:/# exit
  12. exit
  13. [root@wangpengliang ~]# docker container ls
  14. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
  15. a1ea64a6c039 nginx:3.0 "/docker-entrypoint.…" 18 hours ago Up 18 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3
  16. 470e01aed950 nginx "/docker-entrypoint.…" 3 days ago Up 2 minutes 0.0.0.0:80->80/tcp, :::80->80/tcp webserver

注意:如果从这个 stdin 中 exit,会导致容器的停止

exec

docker exec 后边可以跟多个参数,这里主要说明 -i -t 参数

  • 只用 -i 参数时由于没有分配伪终端,界面没有熟悉的 Linux 命令提示符,但命令执行结果仍然可以返回
  • -i -t 参数一起使用时,则可以看到熟悉的 Linux 命令提示符
    1. [root@wangpengliang ~]# docker exec -it 2687f43c0e42 bash
    2. root@2687f43c0e42:/# ls
    3. bin boot dev etc home lib lib32 lib64 libx32 media mnt opt proc root run sbin srv sys tmp usr var
    如果从这个 stdin 中 exit,不会导致容器的停止。这就是为什么推荐使用 docker exec 的原因,更多参数说明使用 docker exec --help 查看。

    导入和导出

    导出容器

    如果要导出本地某个容器,可以使用 docker export 命令,这将导出容器快照到本地文件。
    1. [root@wangpengliang ~]# docker ps -a
    2. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
    3. 2687f43c0e42 ubuntu "/bin/bash" 2 hours ago Up 2 minutes romantic_greider
    4. a1ea64a6c039 nginx:3.0 "/docker-entrypoint.…" 20 hours ago Up 20 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3
    5. bab7ffb6e2e9 nginx:2.0 "/docker-entrypoint.…" 3 days ago Exited (255) 20 hours ago 0.0.0.0:81->80/tcp, :::81->80/tcp webserver2
    6. 470e01aed950 nginx "/docker-entrypoint.…" 3 days ago Up 2 hours 0.0.0.0:80->80/tcp, :::80->80/tcp webserver
    7. [root@wangpengliang ~]# docker export 2687f43c0e42 > ubuntu.tar

    导入容器

    可以使用 docker import 从容器快照文件中再导入为镜像。比如: ```shell [root@wangpengliang ~]# docker ps -a CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES 2687f43c0e42 ubuntu “/bin/bash” 2 hours ago Up 2 minutes romantic_greider a1ea64a6c039 nginx:3.0 “/docker-entrypoint.…” 20 hours ago Up 20 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3 bab7ffb6e2e9 nginx:2.0 “/docker-entrypoint.…” 3 days ago Exited (255) 20 hours ago 0.0.0.0:81->80/tcp, :::81->80/tcp webserver2 470e01aed950 nginx “/docker-entrypoint.…” 3 days ago Up 2 hours 0.0.0.0:80->80/tcp, :::80->80/tcp webserver [root@wangpengliang ~]# docker export 2687f43c0e42 >ubuntu.tar [root@wangpengliang ~]# cat ubuntu.tar|docker import - test/ubuntu:v1.0 sha256:557b6b4bfca63802c6b89bb4c8f1800a2af7954fdff345bd94adba2c770ad4d4 [root@wangpengliang ~]# docker image ls REPOSITORY TAG IMAGE ID CREATED SIZE test/ubuntu v1.0 557b6b4bfca6 9 seconds ago 72.7MB nginx 3.0 f89693105193 20 hours ago 133MB nginx 2.0 546ad28bcf61 3 days ago 133MB redis 6.2.4 fad0ee7e917a 5 days ago 105MB redis latest fad0ee7e917a 5 days ago 105MB nginx latest d1a364dc548d 13 days ago 133MB ubuntu latest 7e0aa2d69a15 6 weeks ago 72.7MB hello-world latest d1165f221234 3 months ago 13.3kB
  1. 此外,也可以通过指定 URL 或者某个目录来导入,比如:
  2. ```shell
  3. docker import http://example.com/exampleimage.tgz example/imagerepo

注:用户既可以使用 docker load 来导入镜像存储文件到本地镜像库,也可以使用 docker import 来导入一个容器快照到本地镜像库。这两者的区别在于容器快照文件将丢弃所有的历史记录和元数据信息(即仅保存容器当时的快照状态),而镜像存储文件将保存完整记录,体积也要大。此外,从容器快照文件导入时可以重新指定标签等元数据信息

删除容器

可以使用 **docker container rm** 来删除一个处于终止状态的容器

  1. [root@wangpengliang ~]# docker ps -a
  2. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
  3. 2687f43c0e42 ubuntu "/bin/bash" 2 hours ago Up 6 minutes romantic_greider
  4. a1ea64a6c039 nginx:3.0 "/docker-entrypoint.…" 20 hours ago Up 20 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3
  5. bab7ffb6e2e9 nginx:2.0 "/docker-entrypoint.…" 3 days ago Exited (255) 20 hours ago 0.0.0.0:81->80/tcp, :::81->80/tcp webserver2
  6. 470e01aed950 nginx "/docker-entrypoint.…" 3 days ago Up 2 hours 0.0.0.0:80->80/tcp, :::80->80/tcp webserver
  7. [root@wangpengliang ~]# docker container rm bab7ffb6e2e9
  8. bab7ffb6e2e9
  9. [root@wangpengliang ~]# docker ps -a
  10. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
  11. 2687f43c0e42 ubuntu "/bin/bash" 2 hours ago Up 7 minutes romantic_greider
  12. a1ea64a6c039 nginx:3.0 "/docker-entrypoint.…" 20 hours ago Up 20 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3
  13. 470e01aed950 nginx "/docker-entrypoint.…" 3 days ago Up 2 hours 0.0.0.0:80->80/tcp, :::80->80/tcp webserver

如果要删除一个运行中的容器,可以添加 **-f** 参数。Docker 会发送 **SIGKILL** 信号给容器

  1. [root@wangpengliang ~]# docker ps -a
  2. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
  3. 2687f43c0e42 ubuntu "/bin/bash" 2 hours ago Up 8 minutes romantic_greider
  4. a1ea64a6c039 nginx:3.0 "/docker-entrypoint.…" 20 hours ago Up 20 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3
  5. 470e01aed950 nginx "/docker-entrypoint.…" 3 days ago Up 2 hours 0.0.0.0:80->80/tcp, :::80->80/tcp webserver
  6. [root@wangpengliang ~]# docker container rm -f 2687f43c0e42
  7. 2687f43c0e42
  8. [root@wangpengliang ~]# docker ps -a
  9. CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
  10. a1ea64a6c039 nginx:3.0 "/docker-entrypoint.…" 20 hours ago Up 20 hours 0.0.0.0:82->80/tcp, :::82->80/tcp webserver3
  11. 470e01aed950 nginx "/docker-entrypoint.…" 3 days ago Up 2 hours 0.0.0.0:80->80/tcp, :::80->80/tcp webserver

清理所有处于终止状态的容器
docker container ls -a 命令可以查看所有已经创建的包括终止状态的容器,如果数量太多一个个删除可能会很麻烦,用下面的命令可以清理掉所有处于终止状态的容器。

  1. $ docker container prune