diff --git a/image/list.md b/image/list.md index 989a4a6..4b63188 100644 --- a/image/list.md +++ b/image/list.md @@ -1,29 +1,143 @@ -## 列出本地镜像 -使用 `docker images` 显示本地已有的镜像。 +## 列出镜像 + +要想列出已经下载下来的镜像,可以使用 `docker images` 命令。 + +```bash +$ docker images +REPOSITORY TAG IMAGE ID CREATED SIZE +redis latest 5f515359c7f8 5 days ago 183 MB +nginx latest 05a60462f8ba 5 days ago 181 MB +mongo 3.2 fe9198c04d62 5 days ago 342 MB + 00285df0df87 5 days ago 342 MB +ubuntu 16.04 f753707788c5 4 weeks ago 127 MB +ubuntu latest f753707788c5 4 weeks ago 127 MB +ubuntu 14.04 1e0c3dd64ccd 4 weeks ago 188 MB ``` -$ sudo docker images -REPOSITORY TAG IMAGE ID CREATED VIRTUAL SIZE -ubuntu 12.04 74fe38d11401 4 weeks ago 209.6 MB -ubuntu precise 74fe38d11401 4 weeks ago 209.6 MB -ubuntu 14.04 99ec81b80c55 4 weeks ago 266 MB -ubuntu latest 99ec81b80c55 4 weeks ago 266 MB -ubuntu trusty 99ec81b80c55 4 weeks ago 266 MB + +列表包含了仓库名、标签、镜像 ID、创建时间以及所占用的空间。 + +其中仓库名、标签在之前的基础概念章节已经介绍过了。**镜像 ID** 则是镜像的唯一标识,一个镜像可以对应多个**标签**。因此,在上面的例子中,我们可以看到 `ubuntu:16.04` 和 `ubuntu:latest` 拥有相同的 ID,因为它们对应的是同一个镜像。 + +### 镜像体积 + +如果仔细观察,会注意到,这里标识的所占用空间和在 Docker Hub 上看到的镜像大小不同。比如,`ubuntu:16.04` 镜像大小,在这里是 `127 MB`,但是在 [Docker Hub](https://hub.docker.com/r/library/ubuntu/tags/) 显示的却是 `50 MB`。这是因为 Docker Hub 中显示的体积是压缩后的体积。在镜像下载和上传过程中镜像是保持着压缩状态的,因此 Docker Hub 所显示的大小是网络传输中更关心的流量大小。而 `docker images` 显示的是镜像下载到本地后,展开的大小,准确说,是展开后的各层所占空间的总和,因为镜像到本地后,查看空间的时候,更关心的是本地磁盘空间占用的大小。 + +另外一个需要注意的问题是,`docker images` 列表中的镜像体积总和并非是所有镜像实际硬盘消耗。由于 Docker 镜像是多层存储结构,并且可以继承、复用,因此不同镜像可能会因为使用相同的基础镜像,从而拥有共同的层。由于 Docker 使用 Union FS,相同的层只需要保存一份即可,因此实际镜像硬盘占用空间很可能要比这个列表镜像大小的总和要小的多。 + +### 虚悬镜像 + +上面的镜像列表中,还可以看到一个特殊的镜像,这个镜像既没有仓库名,也没有标签,均为 ``。: + +```bash + 00285df0df87 5 days ago 342 MB +``` + +这个镜像原本是有镜像名和标签的,原来为 `mongo:3.2`,随着官方镜像维护,发布了新版本后,重新 `docker pull mongo:3.2` 时,`mongo:3.2` 这个镜像名被转移到了新下载的镜像身上,而旧的镜像上的这个名称则被取消,从而成为了 ``。除了 `docker pull` 可能导致这种情况,`docker build` 也同样可以导致这种现象。由于新旧镜像同名,旧镜像名称被取消,从而出现仓库名、标签均为 `` 的镜像。这类无标签镜像也被称为 **虚悬镜像(dangling image)** ,可以用下面的命令专门显示这类镜像: + +```bash +$ docker images -f dangling=true +REPOSITORY TAG IMAGE ID CREATED SIZE + 00285df0df87 5 days ago 342 MB +``` + +一般来说,虚悬镜像已经失去了存在的价值,是可以随意删除的,可以用下面的命令删除。 + +```bash +$ docker rmi $(docker images -q -f dangling=true) +``` + +### 中间层镜像 + +为了加速镜像构建、重复利用资源,Docker 会利用 **中间层镜像**。所以在使用一段时间后,可能会看到一些依赖的中间层镜像。默认的 `docker images` 列表中只会显示顶层镜像,如果希望显示包括中间层镜像在内的所有镜像的话,需要加 `-a` 参数。 + +```bash +$ docker images -a +``` + +这样会看到很多无标签的镜像,与之前的虚悬镜像不同,这些无标签的镜像很多都是中间层镜像,是其它镜像所依赖的镜像。这些无标签镜像不应该删除,否则会导致上层镜像因为依赖丢失而出错。实际上,这些镜像也没必要删除,因为之前说过,相同的层只会存一遍,而这些镜像是别的镜像的依赖,因此并不会因为它们被列出来而多存了一份,无论如何你也会需要它们。只要删除那些依赖它们的镜像后,这些依赖的中间层镜像也会被连带删除。 + +### 列出部分镜像 + +不加任何参数的情况下,`docker images` 会列出所有顶级镜像,但是有时候我们只希望列出部分镜像。`docker images` 有好几个参数可以帮助做到这个事情。 + +根据仓库名列出镜像 + +```bash +$ docker images ubuntu +REPOSITORY TAG IMAGE ID CREATED SIZE +ubuntu 16.04 f753707788c5 4 weeks ago 127 MB +ubuntu latest f753707788c5 4 weeks ago 127 MB +ubuntu 14.04 1e0c3dd64ccd 4 weeks ago 188 MB +``` + +列出特定的某个镜像,也就是说指定仓库名和标签 + +```bash +$ docker images ubuntu:16.04 +REPOSITORY TAG IMAGE ID CREATED SIZE +ubuntu 16.04 f753707788c5 4 weeks ago 127 MB +``` + +除此以外,`docker images` 还支持强大的过滤器参数 `--filter`,或者简写 `-f`。之前我们已经看到了使用过滤器来列出虚悬镜像的用法,它还有更多的用法。比如,我们希望看到在 `mongo:3.2` 之后建立的镜像,可以用下面的命令: + +```bash +$ docker images -f since=mongo:3.2 +REPOSITORY TAG IMAGE ID CREATED SIZE +redis latest 5f515359c7f8 5 days ago 183 MB +nginx latest 05a60462f8ba 5 days ago 181 MB +``` + +想查看某个位置之前的镜像也可以,只需要把 `since` 换成 `before` 即可。 + +此外,如果镜像构建时,定义了 `LABEL`,还可以通过 `LABEL` 来过滤。 + +```bash +$ docker images -f label=com.example.version=0.1 ... ``` -在列出信息中,可以看到几个字段信息 +### 以特定格式显示 -* 来自于哪个仓库,比如 ubuntu -* 镜像的标记,比如 14.04 -* 它的 `ID` 号(唯一) -* 创建时间 -* 镜像大小 +默认情况下,`docker images` 会输出一个完整的表格,但是我们并非所有时候都会需要这些内容。比如,刚才删除虚悬镜像的时候,我们需要利用 `docker images` 把所有的虚悬镜像的 ID 列出来,然后才可以交给 `docker rmi` 命令作为参数来删除指定的这些镜像,这个时候就用到了 `-q` 参数。 -其中镜像的 `ID` 唯一标识了镜像,注意到 `ubuntu:14.04` 和 `ubuntu:trusty` 具有相同的镜像 `ID`,说明它们实际上是同一镜像。 - -`TAG` 信息用来标记来自同一个仓库的不同镜像。例如 `ubuntu` 仓库中有多个镜像,通过 `TAG` 信息来区分发行版本,例如 `10.04`、`12.04`、`12.10`、`13.04`、`14.04` 等。例如下面的命令指定使用镜像 `ubuntu:14.04` 来启动一个容器。 -``` -$ sudo docker run -t -i ubuntu:14.04 /bin/bash +```bash +$ docker images -q +5f515359c7f8 +05a60462f8ba +fe9198c04d62 +00285df0df87 +f753707788c5 +f753707788c5 +1e0c3dd64ccd ``` -如果不指定具体的标记,则默认使用 `latest` 标记信息。 +`--filter` 配合 `-q` 产生出指定范围的 ID 列表,然后送给另一个 `docker` 命令作为参数,从而针对这组实体成批的进行某种操作的做法在 Docker 命令行使用过程中非常常见,不仅仅是镜像,将来我们会在各个命令中看到这类搭配以完成很强大的功能。因此每次在文档看到过滤器后,可以多注意一下它们的用法。 + +另外一些时候,我们可能只是对表格的结构不满意,希望自己组织列;或者不希望有标题,这样方便其它程序解析结果等,这就用到了 [Go 的模板语法](https://gohugo.io/templates/go-templates/)。 + +比如,下面的命令会直接列出镜像结果,并且只包含镜像ID和仓库名: + +```bash +$ docker images --format "{{.ID}}: {{.Repository}}" +5f515359c7f8: redis +05a60462f8ba: nginx +fe9198c04d62: mongo +00285df0df87: +f753707788c5: ubuntu +f753707788c5: ubuntu +1e0c3dd64ccd: ubuntu +``` + +或者打算以表格等距显示,并且有标题行,和默认一样,不过自己定义列: + +```bash +$ docker images --format "table {{.ID}}\t{{.Repository}}\t{{.Tag}}" +IMAGE ID REPOSITORY TAG +5f515359c7f8 redis latest +05a60462f8ba nginx latest +fe9198c04d62 mongo 3.2 +00285df0df87 +f753707788c5 ubuntu 16.04 +f753707788c5 ubuntu latest +1e0c3dd64ccd ubuntu 14.04 +```