这个问题与我是否应该关注过量的、未运行的Docker容器有关。

我想知道如何移除旧容器。docker rm 3e552code34a允许你删除一个,但我已经有很多了。Docker rm——help没有提供选择选项(像all一样,或者通过映像名称)。

也许有一个存放这些容器的目录,我可以很容易地手动删除它们?


当前回答

如果你不喜欢删除所有的容器,你可以用docker ps -f before=<container- id >或用docker ps -f since=<container- id >选择在特定容器之前或之后创建的所有容器

假设您已经开发了您的系统,现在它正在工作,但是还剩下一些容器。您希望删除在该工作版本之前创建的容器。使用docker ps确定工作容器的ID。

删除在其他容器之前创建的容器

docker rm $(docker ps -f before=9c49c11c8d21 -q)

另一个例子。您已经在Docker容器上运行了数据库。您已经开发了在另一个容器上运行的应用程序,现在您有许多不需要的容器。

删除在某个容器之后创建的容器

docker rm $(docker ps -f since=a6ca4661ec7f -q)

Docker将容器存储在Ubuntu中的/var/lib/docker/containers中。我认为额外的容器没有其他危害,但会占用磁盘空间。

其他回答

如果您想自动/定期清理退出的容器,并删除运行中的容器不使用的映像和卷,您可以下载映像meltwater/docker-cleanup。

我在生产中使用这种方法,因为我们每天在多个服务器上部署几次,我不想去每个服务器上清理(那将是一件痛苦的事情)。

运行:

docker run -d -v /var/run/docker.sock:/var/run/docker.sock:rw  -v /var/lib/docker:/var/lib/docker:rw --restart=unless-stopped meltwater/docker-cleanup:latest

默认情况下,它将每30分钟运行一次(或者无论您使用DELAY_TIME=1800选项设置多长时间),并清理退出的容器和图像。

更多细节:Docker Cleanup

更新:从Docker 1.13版(2017年1月发布)开始,您可以发出以下命令来清理停止的容器、未使用的卷、悬空的映像和未使用的网络:

docker system prune

如果你想确保你只删除有退出状态的容器,使用这个:

docker ps -aq -f status=exited | xargs docker rm

类似地,如果你在清理docker的东西,你可以用这样的方式去除未标记的、未命名的图像:

docker images -q --no-trunc -f dangling=true | xargs docker rmi

另一种方法,我从Guillaume J. Charmes那里得到的(出处请注明出处)

docker rm `docker ps --no-trunc -aq`

将以优雅的方式移除所有容器。

由Bartosz Bilicki为Windows编写:

FOR /f "tokens=*" %i IN ('docker ps -a -q') DO docker rm %i

美国小妞PowerShell:

docker rm @(docker ps -aq)

Docker 1.13(2016年第四季度)的更新,归功于VonC(稍后在本帖子中):

docker系统修剪将删除所有未使用的数据(即,按顺序:容器停止,卷没有容器和图像没有容器)。

请参阅PR 26108和commit 86de7c0,它们引入了一些新命令,以帮助方便地可视化Docker守护进程数据在磁盘上占用了多少空间,并允许轻松清理“不需要的”多余空间。

docker system prune

WARNING! This will remove:
    - all stopped containers
    - all volumes not used by at least one container
    - all images without at least one container associated to them
Are you sure you want to continue? [y/N] y
#!/bin/bash
# docker-gc --- Remove stopped docker containers

RUNNING=$(docker ps -q)
ALL=$(docker ps -a -q)

for container in $ALL ; do
    [[ "$RUNNING" =~ "$container" ]] && continue
    echo Removing container: $(docker rm $container)
done

因此,我个人建议将此作为映像和容器的部署脚本的一部分,只保留最近的n个容器和映像。我用与git标签相同的版本控制模式标记Docker映像,并且总是用“latest”标记最新的Docker映像。这意味着在没有清理任何东西的情况下,我的Docker映像最终看起来像这样:

REPOSITORY              TAG       IMAGE ID        CREATED         VIRTUAL SIZE
some_repo/some_image    0.0.5     8f1a7c7ba93c    23 hours ago    925.4 MB
some_repo/some_image    latest    8f1a7c7ba93c    23 hours ago    925.4 MB
some_repo/some_image    0.0.4     0beabfa514ea    45 hours ago    925.4 MB
some_repo/some_image    0.0.3     54302cd10bf2    6 days ago      978.5 MB
some_repo/some_image    0.0.2     0078b30f3d9a    7 days ago      978.5 MB
some_repo/some_image    0.0.1     sdfgdf0f3d9a    8 days ago      938.5 MB

现在,当然我不想让我所有的图像(或容器)在我所有的生产盒上永久保存。我只需要最后的3或4个回滚来去掉其他的东西。尤尼克斯的尾巴是你最好的朋友。由于docker images和docker ps都是按日期排序的,我们可以使用tail选择除前三个之外的所有图像并删除它们:

docker rmi $(docker images -q | tail -n +4)

将其与部署脚本一起运行(或在本地运行),以始终保留足够的映像,以便能够轻松地回滚,而不会占用太多空间或将旧映像弄得乱七八糟。

就我个人而言,我只在我的生产盒上保留一个容器,但如果你想要更多的容器,你可以对容器做同样的事情:

docker rm $(docker ps -aq | tail -n +4)

最后,在我的简化示例中,我们一次只处理一个存储库,但是如果您有更多的存储库,您可以使用相同的思想来变得更复杂一些。假设我只想保留some_repo/some_image中的最后三张图像。我只需要混合grep和awk就可以了:

docker rmi $(docker images -a | grep 'some_repo/some_image' | awk '{print $3}' | tail -n +4)

同样,同样的想法也适用于容器,但到这里你已经明白了,所以我就不举例子了。