愛在花開的季節 发表于 2024-7-18 06:23:37

6.Dockerfile及Dockerfile常用指令

Dockerfile是构建docker镜像的脚本文件
Dockerfile有很多的指令构成,指令由上到下依次运行。
每一条指令就是一层镜像,层越多,体积就越大,启动速度也越慢
井号开头的行是解释行。指令写大写写小写都行,但一般都写为大写。每一行中间都可以有若干空行
在有的github项目中会给你Dockerfile,以便你能更方便的配环境
可以使用docker build将Dockerfile构建为镜像,命令为 docker build -t [镜像名称] .


[*]关于docker build的具体使用方法在 4.docker镜像及相关命令 有提到
目录
1  引用 FROM
2  复制 ADD
3  复制 COPY
4  定义维护者信息 MAINTAINER
5  定义元数据 LABEL
6  定义工作目录 WORKDIR
7  定义变量 ENV
8  实行命令 RUN
9  打开容器后实行的语句 CMD
9.1  ls命令简介
9.2  中括号写法
9.2.1  多个参数
9.2.2  参数和值
9.3  直接写
9.4  CMD给ENTRYPOINT提供参数
10  打开后容器实行的语句 ENTRYPOINT
11  定义变量 ARG
12  子镜像中要做的事变 ONBUILD
12.1  镜像的父子关系
12.2  简单使用
13  准备暴露的端口 EXPOSE
14  挂载数据卷 VOLUME
14.1  中括号
14.1.1  直接启动
14.1.2  加-v启动
14.2  直接写

1  引用 FROM

FROM是引用底子镜像,底子镜像就是官方或者别的做好的,我们一般站在巨人的肩膀上添加新的功能
在hello world的例子中,hello world的镜像就是基于scratch镜像的再创作


[*]默认情况下FROM会先从本地拉取,假如本地没有就会到源拉取
https://i-blog.csdnimg.cn/direct/7c45a982ce1f413192c541a9c839c750.png
scratch是空的镜像,相当于面向对象编程中的基类。scratch只在Dockerfile中继承,不能通过pull拉取,不能run,没有tag
镜像不必须带FROM,不带FROM的镜像叫做底子镜像。好比scratch镜像的第一句就不是FROM scratch
我们本身写Dockerfile的时间基本都要带FROM,在别人的镜像上就行修改
FROM引用的是旧层,不产生新层
2  复制 ADD

ADD不太好用,还是用COPY要更好用一点
从Dockerfile文件地点的呆板 复制文件到 镜像中。在hello world的例子中是将 hello 这个可实行文件(这里用的是相对路径[相对路径指的是相对Dockerfile文件的路径],也可以使用绝对路径),复制到镜像的 / 位置(根路径)
https://i-blog.csdnimg.cn/direct/6060ff81e6a347eb82d070eef7bb08f4.png
使用ADD指令,假如将可实行文件hello更换成一个压缩文件,压缩文件复制到容器后会自动解压
使用ADD指令,假如将可实行文件hello更换成一个url,url会自动下载到容器的指定目录中(相当于wget)
假如将可实行文件hello更换成一个文件夹(文件夹末了必须要加上斜杠),那么就会将文件夹中全部内容复制到容器的指定位置
3  复制 COPY

COPY的功能与ADD相似,同样是从Dockerfile文件地点的呆板 复制文件到 镜像中。
与ADD的区别为


[*]使用COPY从复制压缩包后不会自动解压
[*]使用COPY不能复制url
4  定义维护者信息 MAINTAINER



[*]官方不发起使用MAINTAINER指令,但一些老的Dockerfile中会有MAINTAINER,能看懂就行
现实就是写创作者的名字,我简单做个例子
https://i-blog.csdnimg.cn/direct/78d1e0b251604b82ab3e19c572f4fd35.png
然后我们build一下
https://i-blog.csdnimg.cn/direct/9a73797e2e044e43a122fc4f2a8c444e.png
build之后可以看到MAINTAINER写的内容
https://i-blog.csdnimg.cn/direct/e91b03aee83647a2a0af6ac45da4ca3d.png
5  定义元数据 LABEL

LABEL写什么东西都可以,我们简单做个例子
https://i-blog.csdnimg.cn/direct/570fe8fdee1d49ca8967edc3b03cf3d0.png
https://i-blog.csdnimg.cn/direct/eda1f1a966c148fbb77bf8b287b50e99.png
检察元数据的时间发现Author并没有被覆盖
https://i-blog.csdnimg.cn/direct/e98003eadce9463ea439864c184be98d.png
而是把LABEL的内容都写在Labels中了
https://i-blog.csdnimg.cn/direct/5dfbfe813f4341debfaddcf808aea24c.png
每个镜像层都由 镜像文件系统 和 镜像json文件 两部分构成。LABEL命令虽然没有改变镜像文件系统,但是改变了镜像的json文件,所以LABEL也会产生新的一层
6  定义工作目录 WORKDIR

我们创建一个ubuntu的容器,然后开启容器,发现工作目录默认在 / 这个位置
https://i-blog.csdnimg.cn/direct/a3cfaf7427944dbfaa1d7201c97b5351.png
我们可以更改操纵目录,好比我们想将工作目录搞到 /home 下
https://i-blog.csdnimg.cn/direct/f42ed7161f2e417a954d6741f983840d.png
进入后发现工作目录是/home
https://i-blog.csdnimg.cn/direct/76c5e9005ec04abaa8fe71519ec0343c.png
WORKDIR可以写多个,好比我这样写,那么进去后的目录就为/usr/local。相当于后面是前面的相对路径,而不是覆盖掉前面的路径
https://i-blog.csdnimg.cn/direct/74c51feb2d0a4eb5b63b6c8f89501a31.png
7  定义变量 ENV

还是改变工作目录,这次我们用变量的情势来搞。ENV定义变量,后面使用$来调用变量
https://i-blog.csdnimg.cn/direct/ed72844680274342ab0d29c6a543f5c0.png
可以乐成调用变量
https://i-blog.csdnimg.cn/direct/2600c6b03d404d198e6a8fb76a53b9ba.png
同一行可以写多个,好比
https://i-blog.csdnimg.cn/direct/92f8b9dd9e5a4093b592139cdcce71aa.png
8  实行命令 RUN

在ubuntu镜像中没有ifconfig这个命令,我现在想搞个有ifconfig的镜像
https://i-blog.csdnimg.cn/direct/b7f029c3143a45bfb2c35bc2ac78ac36.png
那么我们需要在镜像创建的时间就安装,需要实行一些命令
https://i-blog.csdnimg.cn/direct/822dff681a854a849aba96953929ae41.png
在构建的时间你就可以看到实行的过程
https://i-blog.csdnimg.cn/direct/6979fd5b473c4af4a7afcc70b02e54f2.png
这样创建的镜像运行后就有ifconfig这个指令了
https://i-blog.csdnimg.cn/direct/8773a856fd274e0abbb351c7d6978684.png
我们不发起RUN分多行写,因为这样会产生多层
https://i-blog.csdnimg.cn/direct/d592db5cc8dd45f8aaddfb8bcc5c53ad.png
我们可以把两行合起来写
https://i-blog.csdnimg.cn/direct/14a168a217d24ecf956d3c69c26935fd.png
这样这个RUN就只有一层了
https://i-blog.csdnimg.cn/direct/fbfcc762b2e04f18a7e91635a4fb916e.png
假如比较长的话影响观感,可以这样分行来写
https://i-blog.csdnimg.cn/direct/581465b1270942ce9150971625b5cef9.png
也可以用下面的语法来写,EXECUTABLE为可实行的东西,后面PARAM1,PARAME2是EXECUTABLE的参数
https://i-blog.csdnimg.cn/direct/84cca3a20e6649978395ce9c8d561b33.png
9  打开容器后实行的语句 CMD

9.1  ls命令简介

我们用ls来验证CMD,假如只输入ls,那么出现的结果是这样的
https://i-blog.csdnimg.cn/direct/ca6a5907c50345039454fe65faf18fa6.png
假如参加参数 -l 那么出现的结果是这样的,我们可以发现结果中有一些指向
https://i-blog.csdnimg.cn/direct/10ae93ad3c8e4adfa1581ebede50056a.png
假如再参加 -a,出现的结果是这样的。我们发现结果中出现了隐蔽文件
https://i-blog.csdnimg.cn/direct/578c062b5e8e49cc83ffa399043dec0b.png
9.2  中括号写法

https://i-blog.csdnimg.cn/direct/004d542fae164dfbb940673580bea3d0.png
9.2.1  多个参数

我们简单做个例子
https://i-blog.csdnimg.cn/direct/89db4332368d48cb85f17d22feac394e.png
发现结果中有指向,并且有隐蔽文件,说明-l与-a见效了
https://i-blog.csdnimg.cn/direct/691feff4c8214ff6ba18a7cbe44c464e.png
9.2.2  参数和值

像ls这种-l,-a这种参数不加任何的值,有的参数可以加值,好比 /bin/bash 的-c参数,-c参数可以加命令,我们简单用一下
https://i-blog.csdnimg.cn/direct/a6332e608b9a46ca9024df159db328bd.png
https://i-blog.csdnimg.cn/direct/86aa80996d6d40daacc680ebbbd6142a.png
也可以在docker ps -a 中检察到运行的命令
https://i-blog.csdnimg.cn/direct/b85d7b0ff6ad4e3a8674d94548bae63a.png
从这里你就看出,相当于是只要空格你就写个逗号,然后凑个数组
9.3  直接写

我们简单写一下
https://i-blog.csdnimg.cn/direct/51d759b732d74912ab9393cbd7d8a296.png
https://i-blog.csdnimg.cn/direct/a727b6627d0e417897b0bad57231b58c.png
可以通过docker ps -a看一下
https://i-blog.csdnimg.cn/direct/8f121678054b4765a192d16471bc0651.png
9.4  CMD给ENTRYPOINT提供参数

第三种是提供给ENTRYPOINT的参数。假如CMD不是为ENTRYPOINT提供参数,那么不发起ENTRYPOINT与CMD同时出现
假如使用CMD给ENTRYPOINT添加参数,ENTRYPOINT必须为中括号的写法。我们简单用一下
https://i-blog.csdnimg.cn/direct/ccd2f2c0b53d4a0a849bd06d45daed84.png


[*]CMD与ENTRYPOINT谁写前面谁写后面都行
发现可以达到结果
https://i-blog.csdnimg.cn/direct/aa3eea00230542e2a9bec75dfd07a4b8.png
由于docker run的COMMAND这个参数本质上是覆盖CMD,所以可以在docker run中给ENTRYPOINT参数,好比
https://i-blog.csdnimg.cn/direct/acb71d05f5e145c78a5cf7bd8b0e9298.png
相当于把 -l -s 更换成了 -s
https://i-blog.csdnimg.cn/direct/108beaf6e59e45f58ed419aa81f4db14.png
10  打开后容器实行的语句 ENTRYPOINT

与CMD的区别为CMD可以通过docker run 的参数替代,但是ENTRYPOINT不会被替代,也就是说运行容器后怎么都会实行一次ENTRYPOINT的内容
docker run可以定义CMD与参数,Dockerfile中的CMD也可以定义CMD与参数,但你不能用docker run的CMD配Dockerfile的参数,也不能用Dockerfile的CMD配docker run 的参数。好比你Dockerfile中有CMD,然后docker run中只给参数,这样是不对的
https://i-blog.csdnimg.cn/direct/e43b491bd3ce422b85d35bb19595bd42.png
ENTRYPOINT本身两种写法,一种带中括号
https://i-blog.csdnimg.cn/direct/8856781b9ede43fab299f71f859819e7.png
一种不带中括号
https://i-blog.csdnimg.cn/direct/82ebf359eed041abb3d8e0610188d72b.png
用法与CMD同等,就不举例子了
ENTRYPOINT与CMD指令会将启动命令写在json文件中,改动了json文件从而会产生新的镜像层
11  定义变量 ARG

ENV的值不能被build的参数 --build-arg 覆盖,但是ARG可以
我们简单做个例子
https://i-blog.csdnimg.cn/direct/0a5e645c17fe4d5c902cb12d13101de2.png
build的时间发现这两个变量都可以调用
https://i-blog.csdnimg.cn/direct/c50891c789bc46ccbb67f05c6f72914d.png
我们此时尝试对name与age这两个变量进行覆盖
https://i-blog.csdnimg.cn/direct/de2f842d7c564689aea12c107a458689.png
发现name(ENV定义的变量)不能覆盖,age(ARG定义的变量)可以被覆盖
https://i-blog.csdnimg.cn/direct/acb926c7577a414b8913925ee301286e.png
一个ARG命令只能定义一个变量,假如要定义多个变量需要多个ARG
12  子镜像中要做的事变 ONBUILD

https://i-blog.csdnimg.cn/direct/4cab176a52314b8ba476ec67361fb72e.png
12.1  镜像的父子关系

在helloworld的例子中,我们通过Dockerfile创建的helloworld镜像 的 父镜像 是 scratch。helloworld镜像称为scratch的子镜像
https://i-blog.csdnimg.cn/direct/d755831bb95545289c2212b7bbff103b.png
假如B镜像的Dockerfile通过FROM使用了helloworld镜像,那么B镜像为helloworld镜像的子镜像
12.2  简单使用

我现在Dockerfile内容如下
https://i-blog.csdnimg.cn/direct/a43849e380e04c1e8bf27afb3965285f.png
build之后run,发现ifconfig用不了
https://i-blog.csdnimg.cn/direct/442f32e0becc43dbbf96ef38271f2b0e.png
我们此时再搞一个如下的Dockerfile将其定名为son
https://i-blog.csdnimg.cn/direct/7866dc11af9245c4968f38161bc1cca7.png
然后build->run->ifconfig
https://i-blog.csdnimg.cn/direct/f9b820a05fd541079a56f15bd2707dab.png
发现可以使用,这个就可以证明 ONBUILD 不是构建本身镜像时间做的事变,而是构建子镜像做的事变
13  准备暴露的端口 EXPOSE

EXPOSE这个参数是给人看的,不是给呆板看的。EXPOSE参数对天生镜像没有作用,暴露端口的时间依然要用 -p
我们简单用一下
https://i-blog.csdnimg.cn/direct/da35eb6566f44af0bc0c813b854cd860.png
14  挂载数据卷 VOLUME

VOLUME后接的是容器内的挂载点,可以是一个可以是多个
一般不使用VOLUME,因为使用VOLUME只能通过-v覆盖的方式自定义宿主机内的挂载点。
14.1  中括号

好比我在容器中创建 /home/A 和 /home/B 这两个挂载点
https://i-blog.csdnimg.cn/direct/3a77703cd0314263a53f8756daeab0ce.png
把上面的dockerfile创建为镜像
https://i-blog.csdnimg.cn/direct/8c17917245a2489ea719982c78928061.png
https://i-blog.csdnimg.cn/direct/be774410e2f84c36bc2a767946ea47df.png
14.1.1  直接启动

我们先直接启动该镜像
https://i-blog.csdnimg.cn/direct/c506082a454d4ceebcc4a5ad678c8342.png
然后检察这个镜像的信息
https://i-blog.csdnimg.cn/direct/d85e8dbacf4d48bf9eab34d4573bb8a9.png
发现数据卷的位置默认在/var/lib这个里面
https://i-blog.csdnimg.cn/direct/0b2b24c381bb4cc0a3b8ff7dac304cfc.png
14.1.2  加-v启动

加-v会多一组数据卷,不会影响之前的
https://i-blog.csdnimg.cn/direct/256cf13cdcb84717ae109ad1c0a754b0.png
https://i-blog.csdnimg.cn/direct/30c405e47cd347bf8336edca810a8978.png
假如容器内的挂载点相同就会覆盖(不测了,一般不会这么干)
14.2  直接写

好比还是在容器中创建 /home/A与/home/B 这两个挂载点
https://i-blog.csdnimg.cn/direct/ebbb4696a2df4c0a9c3f7393e38d5975.png
之后创建镜像,创建容器,发现里面有A和B两个挂载点
https://i-blog.csdnimg.cn/direct/6642a1fd3baf4bfbab37e0c4f9396f4a.png
宿主机的挂载点在 /var/lib/docker/volumes 中,里面长名字的文件夹,代表差别的挂载点。可以根据创建的时间大概推测出来新创建的容器挂载点是哪个。这里面有两个文件夹分别代表A和B,我们先选一个看一下
https://i-blog.csdnimg.cn/direct/531bf34928da4c4da138de52ded1fe79.png
进入这个文件需要输入密码(包罗之进步入 /var/lib/docker 这一级也需要密码)
https://i-blog.csdnimg.cn/direct/885621f1a0534290b71a760f136ddb6e.png
现在这个里面是空的
https://i-blog.csdnimg.cn/direct/1daae2339a0f47f8b523ecaf1615ac0e.png
这个时间我在A里面创建一个文件夹
https://i-blog.csdnimg.cn/direct/8e35e8a9123e4ceeae3778d37363e80d.png
刷新宿主机发现能同步
https://i-blog.csdnimg.cn/direct/c772d74de78945f29d70109d8b408cf7.png
那么后面再找B的对应文件夹,估计是这个
https://i-blog.csdnimg.cn/direct/332eb95e6422450cb66488b14cfdc5bb.png
在这里创建一个名为1的文件夹
https://i-blog.csdnimg.cn/direct/326ce7b37fb74644b00ac25db87fa4af.png
发现可以同步
https://i-blog.csdnimg.cn/direct/8445fb25c7414a3fac47a3167c778c0e.png


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: 6.Dockerfile及Dockerfile常用指令