Docker 第二章 使用镜像

一. 获取镜像

  从 Docker 镜像仓库获取镜像的命令是 docker pull。其命令格式为:

docker pull [选项] [Docker Registry 地址[:端口号]/]仓库名[:标签]

具体的选项可以通过 docker pull --help 命令看到,这里我们说一下镜像名称的格式。

  • Docker 镜像仓库地址:地址的格式一般是 <域名/IP>[:端口号]。默认地址是 Docker Hub。
  • 仓库名:如之前所说,这里的仓库名是两段式名称,即 <用户名>/<软件名>。对于 Docker Hub,如果不给出用户名,则默认为 library,也就是官方镜像。
 
例如:
#docker search  busybox  搜索仓库中的对应镜像
[root@localhost ~]# docker search busybox
NAME                        DESCRIPTION                                     STARS               OFFICIAL            AUTOMATED
busybox                     Busybox base image.                             1583   
 
#docker pull 下载镜像
[root@localhost ~]# docker pull progrium/busybox
Using default tag: latest
latest: Pulling from progrium/busybox
b0dc45cd432d: Download complete
364328af40b6: Download complete 
9c7abf28af64: Download complete 
635bab23d5f1: Download complete 
054e7786c1b6: Download complete 
5100e35a43b2: Download complete 
Digest: sha256:438fd20dc8664ce7c253e65079c08006aa52684314a83722c7f1834188119ad4
Status: Downloaded newer image for progrium/busybox:latest
 
运行
[root@localhost /]# docker run -it --rm progrium/busybox 
/ # 
/ # ls
bin      etc      lib      linuxrc  mnt      proc     run      sys      usr
dev      home     lib64    media    opt      root     sbin     tmp      var
/ # cat /etc/hostname 
402bd5b6f54a
/ # 
/ # ls

  / # cat /etc/os-release
      NAME=Buildroot
      VERSION=2014.02
      ID=buildroot
      VERSION_ID=2014.02
      PRETTY_NAME="Buildroot 2014.02"

 
  • -it:这是两个参数,一个是 -i:交互式操作,一个是 -t 终端。我们这里打算进入 bash 执行一些命令并查看返回结果,因此我们需要交互式终端。
  • --rm:这个参数是说容器退出后随之将其删除。默认情况下,为了排障需求,退出的容器并不会立即删除,除非手动 docker rm。我们这里只是随便执行个命令,看看结果,不需要排障和保留结果,因此使用 --rm 可以避免浪费空间。

列出镜像

要想列出已经下载下来的镜像,可以使用 docker image ls 命令。

[root@localhost ~]# docker images list
REPOSITORY          TAG                 IMAGE ID            CREATED             SIZE
[root@localhost ~]# 
[root@localhost ~]# 
[root@localhost ~]# docker image list
REPOSITORY                         TAG                 IMAGE ID            CREATED             SIZE
dockerpracticecn/docker_practice   latest              b6bfd54275de        7 days ago          41.8MB
nginx                              latest              53f3fd8007f7        13 days ago         109MB
hello-world                        latest              fce289e99eb9        4 months ago        1.84kB
progrium/busybox                   latest              a67699e37dbd        7 months ago        4.8MB
[root@localhost ~]# 

#列表包含了 仓库名标签镜像 ID创建时间 以及 所占用的空间
 

Docker Hub 中显示的体积是压缩后的体积。在镜像下载和上传过程中镜像是保持着压缩状态的,因此 Docker Hub 所显示的大小是网络传输中更关心的流量大小。而 docker image ls 显示的是镜像下载到本地后,展开的大小,准确说,是展开后的各层所占空间的总和,因为镜像到本地后,查看空间的时候,更关心的是本地磁盘空间占用的大小。

另外一个需要注意的问题是,docker image ls 列表中的镜像体积总和并非是所有镜像实际硬盘消耗。由于 Docker 镜像是多层存储结构,并且可以继承、复用,因此不同镜像可能会因为使用相同的基础镜像,从而拥有共同的层。由于 Docker 使用 Union FS,相同的层只需要保存一份即可,因此实际镜像硬盘占用空间很可能要比这个列表镜像大小的总和要小的多。

你可以通过以下命令来便捷的查看镜像、容器、数据卷所占用的空间。

[root@localhost ~]# docker system df
TYPE                TOTAL               ACTIVE              SIZE                RECLAIMABLE
Images              4                   3                   156MB               4.797MB (3%)
Containers          3                   1                   4B                  2B (50%)
Local Volumes       0                   0                   0B                  0B
Build Cache         0                   0                   0B                  0B
[root@localhost ~]# 

中间层镜像

为了加速镜像构建、重复利用资源,Docker 会利用 中间层镜像。所以在使用一段时间后,可能会看到一些依赖的中间层镜像。默认的 docker image ls 列表中只会显示顶层镜像,如果希望显示包括中间层镜像在内的所有镜像的话,需要加 -a 参数

docker image ls -a

这样会看到很多无标签的镜像,与之前的虚悬镜像不同,这些无标签的镜像很多都是中间层镜像,是其它镜像所依赖的镜像。这些无标签镜像不应该删除,否则会导致上层镜像因为依赖丢失而出错。实际上,这些镜像也没必要删除,因为之前说过,相同的层只会存一遍,而这些镜像是别的镜像的依赖,因此并不会因为它们被列出来而多存了一份,无论如何你也会需要它们。只要删除那些依赖它们的镜像后,这些依赖的中间层镜像也会被连带删除。

列出部分镜像

不加任何参数的情况下,docker image ls 会列出所有顶级镜像,但是有时候我们只希望列出部分镜像。docker image ls 有好几个参数可以帮助做到这个事情。

根据仓库名列出镜像

[root@localhost ~]# docker image ls nginx
REPOSITORY          TAG                 IMAGE ID            CREATED             SIZE
nginx               latest              53f3fd8007f7        13 days ago         109MB
[root@localhost ~]# 
Flag shorthand -h has been deprecated, please use --help

Usage:    docker image ls [OPTIONS] [REPOSITORY[:TAG]]

List images

Aliases:
  ls, images, list

Options:
  -a, --all             Show all images (default hides intermediate images)
      --digests         Show digests
  -f, --filter filter   Filter output based on conditions provided
      --format string   Pretty-print images using a Go template
      --no-trunc        Don't truncate output
  -q, --quiet           Only show numeric IDs
[root@localhost ~]# 

删除本地镜像

 如果要删除本地镜像,可以使用 docker iamge rm 命令,其格式为:

 docker image rm [选项] <镜像1> [<镜像2> ...]
#其中,<镜像> 可以是 镜像短 ID镜像长 ID镜像名 或者 镜像摘要

 我们可以用镜像的完整 ID,也称为 长 ID,来删除镜像。使用脚本的时候可能会用长 ID,但是人工输入就太累了,所以更多的时候是用 短 ID 来删除镜像。docker image ls 默认列出的就已经是短 ID 了,一般取前3个字符以上,只要足够区分于别的镜像就可以了。

[root@localhost ~]# docker image rm hello-world
Error response from daemon: conflict: unable to remove repository reference "hello-world" (must force) - container a14868c3b1ec is using its referenced image fce289e99eb9
[root@localhost ~]# docker image rm hello-world -f      #指定强制删除
Untagged: hello-world:latest
Untagged: hello-world@sha256:92695bc579f31df7a63da6922075d0666e565ceccad16b59c3374d2cf4e8e50e
Deleted: sha256:fce289e99eb9bca977dae136fbe2a82b6b7d4c372474c9235adc1741675f587e
[root@localhost ~]# 

Untagged 和 Deleted


如果观察上面这几个命令的运行输出信息的话,你会注意到删除行为分为两类,一类是 Untagged,另一类是 Deleted。我们之前介绍过,镜像的唯一标识是其 ID 和摘要,而一个镜像可以有多个标签。

因此当我们使用上面命令删除镜像的时候,实际上是在要求删除某个标签的镜像。所以首先需要做的是将满足我们要求的所有镜像标签都取消,这就是我们看到的 Untagged 的信息。因为一个镜像可以对应多个标签,因此当我们删除了所指定的标签后,可能还有别的标签指向了这个镜像,如果是这种情况,那么 Delete 行为就不会发生。所以并非所有的 docker image rm 都会产生删除镜像的行为,有可能仅仅是取消了某个标签而已。

当该镜像所有的标签都被取消了,该镜像很可能会失去了存在的意义,因此会触发删除行为。镜像是多层存储结构,因此在删除的时候也是从上层向基础层方向依次进行判断删除。镜像的多层结构让镜像复用变动非常容易,因此很有可能某个其它镜像正依赖于当前镜像的某一层。这种情况,依旧不会触发删除该层的行为。直到没有任何层依赖当前层时,才会真实的删除当前层。这就是为什么,有时候会奇怪,为什么明明没有别的标签指向这个镜像,但是它还是存在的原因,也是为什么有时候会发现所删除的层数和自己 docker pull 看到的层数不一样的源。

除了镜像依赖以外,还需要注意的是容器对镜像的依赖。如果有用这个镜像启动的容器存在(即使容器没有运行),那么同样不可以删除这个镜像。之前讲过,容器是以镜像为基础,再加一层容器存储层,组成这样的多层存储结构去运行的。因此该镜像如果被这个容器所依赖的,那么删除必然会导致故障。如果这些容器是不需要的,应该先将它们删除,然后再来删除镜像。

用 docker image ls 命令来配合


像其它可以承接多个实体的命令一样,可以使用 docker image ls -q 来配合使用 docker image rm,这样可以成批的删除希望删除的镜像。我们在“镜像列表”章节介绍过很多过滤镜像列表的方式都可以拿过来使用。

 
[root@localhost ~]# docker image rm $(docker image ls -q busybox)
Untagged: busybox:latest
Untagged: busybox@sha256:4b6ad3a68d34da29bf7c8ccb5d355ba8b4babcad1f99798204e7abb43e54ee3d
Deleted: sha256:64f5d945efcc0f39ab11b3cd4ba403cc9fefe1fa3613123ca016cf3708e8cafb
Deleted: sha256:d1156b98822dccbb924b4e5fe16465a7ecac8bfc81d726177bed403a8e70c972
[root@localhost ~]# 

docker commit 的语法格式为:

docker commit [选项] <容器ID或容器名> [<仓库名>[:<标签>]]

我们可以用下面的命令将容器保存为镜像:

[root@localhost /]# docker commit --author "zy" --message "test" webserver nginx:v2
sha256:173e28b87308d544c613021291af19a2176c6e7a173c343b727d1e9ea0da511a
[root@localhost /]# 
[root@localhost /]# docker image ls nginx
REPOSITORY          TAG                 IMAGE ID            CREATED              SIZE
nginx               v2                  173e28b87308        About a minute ago   109MB
nginx               latest              53f3fd8007f7        3 weeks ago          109MB
[root@localhost /]# 

运行这个镜像
[root@localhost /]# docker run --name web2 -d -p 81:80 nginx:v2
362391a295efd23396a01c314a6d0d64cb37daa83e6d11f2cea8e9f295730be9
[root@localhost /]# ss -tnl
State       Recv-Q Send-Q                                                  Local Address:Port                                                                 Peer Address:Port              
LISTEN      0      128                                                                 *:22                                                                              *:*                  
LISTEN      0      100                                                         127.0.0.1:25                                                                              *:*                  
LISTEN      0      128                                                                :::4000                                                                           :::*                  
LISTEN      0      128                                                                :::80                                                                             :::*                  
LISTEN      0      128                                                                :::81                                                                             :::*                  
LISTEN      0      128                                                                :::22                                                                             :::*                  
LISTEN      0      100                                                               ::1:25                                                                             :::*                  
[root@localhost /]# 

使用 Dockerfile 定制镜像

Dockerfile 是一个文本文件,其内包含了一条条的 指令(Instruction),每一条指令构建一层,因此每一条指令的内容,就是描述该层应当如何构建。

docker build [选项] <上下文路径/URL/->

  

[root@localhost /]# mkdir mynginx
[root@localhost /]# cd mynginx/
[root@localhost mynginx]# ls
[root@localhost mynginx]# touch Dockerfile
[root@localhost mynginx]# vim Dockerfile

  [root@localhost mynginx]# cat Dockerfile
   FROM nginx
   RUN echo "test" > /usr/share/nginx/html/index.html
  [root@localhost mynginx]#


[root@localhost mynginx]# docker build -t nginx:v3 .
Sending build context to Docker daemon  2.048kB
Step 1/2 : FROM nginx
 ---> 53f3fd8007f7
Step 2/2 : RUN echo "test" > /usr/share/nginx/html/index.html
 ---> Running in d7e6cbdd001a
Removing intermediate container d7e6cbdd001a
 ---> 04ddf23f6f85
Successfully built 04ddf23f6f85
Successfully tagged nginx:v3
[root@localhost mynginx]# docker image ls nginx
REPOSITORY          TAG                 IMAGE ID            CREATED             SIZE
nginx               v3                  04ddf23f6f85        13 seconds ago      109MB
nginx               v2                  173e28b87308        29 minutes ago      109MB
nginx               latest              53f3fd8007f7        3 weeks ago         109MB
[root@localhost mynginx]# 









































































原文地址:https://www.cnblogs.com/zy09/p/10882854.html