文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Dockerfile中的COPY与ADD命令怎么用

2024-04-02 19:55

关注

这篇文章主要介绍了Dockerfile中的COPY与ADD命令怎么用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Dockerfile中的COPY与ADD命令怎么用文章都会有所收获,下面我们一起来看看吧。

build 上下文的概念

在使用 docker build 命令通过 dockerfile 创建镜像时,会产生一个 build 上下文(context)。所谓的 build 上下文就是 docker build 命令的 path 或 url 指定的路径中的文件的集合。在镜像 build 过程中可以引用上下文中的任何文件,比如我们要介绍的 copy 和 add 命令,就可以引用上下文中的文件。

默认情况下 docker build -t testx . 命令中的 . 表示 build 上下文为当前目录。当然我们可以指定一个目录作为上下文,比如下面的命令:

$ docker build -t testx /home/nick/hc

我们指定 /home/nick/hc 目录为 build 上下文,默认情况下 docker 会使用在上下文的根目录下找到的 dockerfile 文件。

copy 和 add 命令不能拷贝上下文之外的本地文件

对于 copy 和 add 命令来说,如果要把本地的文件拷贝到镜像中,那么本地的文件必须是在上下文目录中的文件。其实这一点很好解释,因为在执行 build 命令时,docker 客户端会把上下文中的所有文件发送给 docker daemon。考虑 docker 客户端和 docker daemon 不在同一台机器上的情况,build 命令只能从上下文中获取文件。如果我们在 dockerfile 的 copy 和 add 命令中引用了上下文中没有的文件,就会收到类似下面的错误:

Dockerfile中的COPY与ADD命令怎么用

与 workdir 协同工作

workdir 命令为后续的 run、cmd、copy、add 等命令配置工作目录。在设置了 workdir 命令后,接下来的 copy 和 add 命令中的相对路径就是相对于 workdir 指定的路径。比如我们在 dockerfile 中添加下面的命令:

workdir /app
copy checkredis.py .

然后构建名称为 testx 的容器镜像,并运行一个容器查看文件路径:

Dockerfile中的COPY与ADD命令怎么用

checkredis.py 文件就是被复制到了 workdir /app 目录下。

copy 命令的简单性

如果仅仅是把本地的文件拷贝到容器镜像中,copy 命令是最合适不过的。其命令的格式为:
copy <src> <dest>

除了指定完整的文件名外,copy 命令还支持 go 风格的通配符,比如:

copy check* /testdir/           # 拷贝所有 check 开头的文件
copy check?.log /testdir/       # ? 是单个字符的占位符,比如匹配文件 check1.log

对于目录而言,copy 和 add 命令具有相同的特点:只复制目录中的内容而不包含目录自身。比如我们在 dockerfile 中添加下面的命令:

workdir /app
copy nickdir .

其中 nickdir 目录的结构如下:

Dockerfile中的COPY与ADD命令怎么用

重新构建镜像 testx,运行一个容器并查看 /app 目录下的内容:

Dockerfile中的COPY与ADD命令怎么用

这里只有 file1 和 file2,少了一层目录 nickdir。如果想让 file1 和 file2 还保存在 nickdir 目录中,需要在目标路径中指定这个目录的名称,比如:

workdir /app
copy nickdir ./nickdir

copy 命令区别于 add 命令的一个用法是在 multistage 场景下。关于 multistage 的介绍和用法请参考笔者的《dockerfile 中的 multi-stage》一文。在 multistage 的用法中,可以使用 copy 命令把前一阶段构建的产物拷贝到另一个镜像中,比如:

from golang:1.7.3
workdir /go/src/github.com/sparkdevo/href-counter/
run go get -d -v golang.org/x/net/html
copy app.go    .
run cgo_enabled=0 goos=linux go build -a -installsuffix cgo -o app .
from alpine:latest
run apk --no-cache add ca-certificates
workdir /root/
copy --from=0 /go/src/github.com/sparkdevo/href-counter/app .
cmd ["./app"]

这段代码引用自《dockerfile 中的 multi-stage》一文,其中的 copy 命令通过指定 --from=0 参数,把前一阶段构建的产物拷贝到了当前的镜像中。

add 命令还可以干其它事情

add 命令的格式和 copy 命令相同,也是:

add <src> <dest>

除了不能用在 multistage 的场景下,add 命令可以完成 copy 命令的所有功能,并且还可以完成两类超酷的功能:

•解压压缩文件并把它们添加到镜像中
•从 url 拷贝文件到镜像中

当然,这些功能也让 add 命令用起来复杂一些,不如 copy 命令那么直观。

解压压缩文件并把它们添加到镜像中

如果我们有一个压缩文件包,并且需要把这个压缩包中的文件添加到镜像中。需不需要先解开压缩包然后执行 copy 命令呢?当然不需要!我们可以通过 add 命令一次搞定:

workdir /app
add nickdir.tar.gz .

这应该是 add 命令的最佳使用场景了!

从 url 拷贝文件到镜像中

这是一个更加酷炫的用法!但是在的最佳实践中却强烈建议不要这么用!!docker 官方建议我们当需要从远程复制文件时,最好使用 curl 或 wget 命令来代替 add 命令。原因是,当使用 add 命令时,会创建更多的镜像层,当然镜像的 size 也会更大(下面的两段代码来自 docker 官方文档):

add  /usr/src/things/
run tar -xjf /usr/src/things/big.tar.xz -c /usr/src/things
run make -c /usr/src/things all

如果使用下面的命令,不仅镜像的层数减少,而且镜像中也不包含 big.tar.xz 文件:

run mkdir -p /usr/src/things \
    && curl -sl  \
    | tar -xjc /usr/src/things \
    && make -c /usr/src/things all

好吧,看起来只有在解压压缩文件并把它们添加到镜像中时才需要 add 命令!

加速镜像构建的技巧

在使用 copy 和 add 命令时,我们可以通过一些技巧来加速镜像的 build 过程。比如把那些最不容易发生变化的文件的拷贝操作放在较低的镜像层中,这样在重新 build 镜像时就会使用前面 build 产生的缓存。比如笔者构建镜像时需要用到下面几个文件:

Dockerfile中的COPY与ADD命令怎么用

其中 myhc.py 文件不经常变化,而 checkmongo.py、checkmysql.py 和 checkredis.py 这三个文件则经常变化,那么我们可这样来设计 dockerfile 文件:

workdir /app
copy myhc.py .
copy check* ./

让 copy myhc.py . 单独占据一个镜像层,当 build 过一次后,每次因 checkmongo.py、checkmysql.py 和 checkredis.py 这三个文件变化而导致的重新 build 都不会重新 build copy myhc.py . 镜像层:

Dockerfile中的COPY与ADD命令怎么用

如上图所示,第二步和第三步都没有重新 build 镜像层,而是使用了之前的缓存,从第四步才开始重新 build 了镜像层。当文件 size 比较大且文件的数量又比较多,尤其是需要执行安装等操作时,这样的设计对于 build 速度的提升还是很明显的。所以我们应该尽量选择能够使用缓存的 dockerfile 写法。

关于“Dockerfile中的COPY与ADD命令怎么用”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“Dockerfile中的COPY与ADD命令怎么用”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网行业资讯频道。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-前端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯