在通过dockerfile创建镜像时,每执行一条会改变文件系统的命令,镜像就会增加一层,比如:
FROM ubuntu:22.04
RUN dd if=/dev/zero of=file.txt bs=1M count=100
RUN rm file.txt
执行RUN dd if=/dev/zero of=file.txt bs=1M count=100
时,这条命令会创建一个100MB大小的文件file.txt,此时docker image会增加一层。
随后执行RUN rm file.txt
,docker image中会再增加一层。
此时就会出现一个奇怪的情形,这个镜像中新增了一个100MB的文件,随后又删掉了,按理说镜像的大小应该和ubuntu:22.04镜像的大小相同,但实际上却要比ubuntu:22.04
多出100MB,这是因为docker image中保留了构建过程的每一层。
这和git的提交历史类似,每执行完一条RUN命令,docker中就会进行一次commit,所以虽然删除了file.txt,但历史记录中却依然存在,无法删除。
如果要解决这个问题,可以将两条RUN命令放在一起:
RUN dd if=/dev/zero of=file.txt bs=1M count=100 && \
rm file.txt
这种情况下,此时当RUN dd if=/dev/zero of=file.txt bs=1M count=100 && rm file.txt
执行结束后镜像才会添加新的一层,而这一层的大小为0。
所以在编写dockerfile时,需要特别注意命令的执行顺序和方式,避免镜像太大。