Docker使用(三)Docker底层分析
四、底层分析
1、Docker镜像原理
1.1 commit镜像
docker commit 提交容器成为一个新的副本
# 命令和git原理类似
docker commit -m=“提交的描述信息” -a=“作者” 容器id 目标镜像名:[TAG]
实操:
# 1、启动一个默认tomcat
# 2、发现这个默认的tomcat没有webapps应用,镜像的原因,官方镜像默认是没有的
# 3、自己copy 进去基本文件
# 4、将我们操作的容器通过commit提交为一个镜像
2、Docker容器数据卷
2.1 什么是容器数据卷
将应用和数据打包成一个镜像
需求:数据持久化
Mysql ,容器删除了,删库跑路。需求:数据存储在本地
总结一句话:数据卷,容器的持久化和同步操作!容器也是可以数据共享的。
============================================================================
2.2 使用容器数据卷
方式一:直接使用命令来挂载 -V
docker run -it -v 主机目录: 容器内目录
[root@localhost ~]# docker run -it -v /home/ceshi:/home centos /bin/bash
============================================================================
通过inspect查询容器状态
[root@localhost home]# docker inspect e4dcbd52904f
测试:
============================================================================
2.3 具名和匿名挂载
# 匿名挂载
-v 容器内路径
[root@localhost /]# docker run -d -P --name nginx01 -v /etc/nginx nginx
# 查看所有的volume的情况
[root@localhost /]# docker volume ls
这里发现,这种就是匿名挂载,我们在-v只写了容器内的路径,没有写容器外的路径!
# 具名挂载
# 通过 -v **卷名:**容器内路径
[root@localhost /]# docker run -d -P --name nginx02 -v juming-nginx:/etc/nginx nginx
5ed2d5cbdef110d9484b1fa0017fc83f5ae2d08b84f4628b38d0f383e890ae47
[root@localhost /]# docker volume ls
[root@localhost /]# docker volume inspect juming-nginx
可以看到,所有的docker容器内的卷,没有指定目录的情况下都是在:/var/lib/docker/volumes/XXX/_data,我们通过具名挂载可以方便的找到我们的一个卷,大多数情况下使用:具名挂载。
# 如何区分匿名挂载/具名挂载
-v 容器内路径 # 匿名挂载
-v 卷名:容器内路径 # 具名挂载
-v /宿主机路径:容器内路径 # 指定路径挂载
拓展:
# 通过 -v 容器内路径:ro rw 改变读写权限(这个权限是相对于容器说的)
ro readonly # 只读
rw readwrite # 读写
[root@localhost /]# docker run -d -P --name nginx03 -v juming-nginx:/etc/nginx:rw nginx
# ro 只要看到ro 说明这个路径只能通过宿主机来操作,容器内是无法操作的。
============================================================================
2.4 初始数据卷之Dockerfile
[root@localhost ~]# cd /home
# 建立一个测试文件夹
[root@localhost home]# mkdir docker_test_volume
# 建立一个dockefile1文件,名字可以随机,每个命令就是docker镜像的一层
[root@localhost docker_test_volume]# vim dockefile1
# build
[root@localhost docker_test_volume]# docker build -f dockefile1 -t cheristhuan/centos:1.0 .
# 运行刚刚镜像
[root@localhost docker_test_volume]# docker run -it c9e468f74ddf /bin/bash
3、Dockerfile
3.1 介绍
构建步骤:
1、编写一个dockerfile文件
2、docker build 构建成为一个镜像
3、docker run 运行镜像
4、docker push 发布镜像(DockerHub、阿里云镜像仓库)
============================================================================
3.2 构建过程
基础知识:
1、每个保留关键字(指令)都必须是大写字母;
2、执行从上到下顺序执行;
3、#表示注释
4、每一个指令都会创建提交一个新的镜像层,并提交。
dockerfile是面向开发的,我们以后发布项目,做镜像,就需要编写dockerfile文件,逐渐成为企业交付的标准,必须要掌握。
**步骤:**开发、部署、运维…缺一不可
步骤1、DockerFile:构建文件,定义了一切的步骤,源代码;
步骤2、DockerImages:通过DockerFile构建生成的镜像,最终发布和运行的产品;
步骤3、Docker容器:容器就是镜像运行起来提供服务器
============================================================================
3.3 常见构建命令
FROM # 1、基础镜像,一切从这里开始构建
MAINTAINER # 2、镜像是谁写的(告诉别人,是你创造的它),建议格式:姓名+邮箱
RUN # 3、镜像构建的时候需要运行的命令(在命令前面加上RUN,即你想要让它做什么用)
ADD # 4、步骤:tomcat镜像,这个tomcat压缩包!添加内容(COPY文件,会自动解压的,往它内部放点东西)
WORKDIR # 5、镜像的工作目录
VOLUME # 6、 挂载的目录
EXPOSE # 7、 暴露端口配置
CMD # 8、指定这个容器启动的时候要运行的命令,只有最后一个会生效, 可被替代
ENTRYPOINT # 9、指定这个容器启动的时候要运行的命令,可以追加命令
ONBUILD # 10、当构建一个被继承 DockerFile这个时候就会运行ONBUILD的指令,触发指令
COPY # 11、类似ADD,将文件拷贝到镜像中
ENV # 12、构建的时候设置环境变量
============================================================================
3.4 实战
Docker Hub 中99%的镜像都是 从这个基础镜像过来的 FROM scratch,然后配置需要的软件和配置来进行构建:
4、Docker网络
4.1 理解Docker0
# 清除全部容器和镜像
[root@localhost ~]# docker rm -f $(docker ps -aq)
[root@localhost ~]# docker rmi -f $(docker images -aq)
# 测试 得到一个eth0@if533
[root@localhost ~]# docker run -d -P --name tomcat01 tomcat
[root@localhost ~]# docker exec -it tomcat01 ip addr
[root@localhost ~]# ping 172.17.0.2
是可以ping的通的docker容器内部
# 思考
1、我们每启动一个docker容器,docker就会给docker容器分配一个ip,就会有一个网卡docker0桥接模式,使用的技术是evth-pair技术!
结论:
tomcat01 和 tomcat02是公用的一个路由器,docker0,所有的容器在不指定网络的情况下,都是docker0路由的。
Docker中的所有的网络接口都是虚拟的,虚拟的转发效率高!(内网传递文件),只要容器删除,对应网桥一对就没有了。
============================================================================
4.2 自定义网络
网络模式:
bridge:桥接 docker(默认)
none:不配置网络
host:和宿主机共享网络
container:容器网络连通(用的少,局限很大)
测试:
# 我们直接启动的命令 --net bridge ,而这个就是我们的docker0
docker run -d -P --name tomcat01 tomcat
docker run -d -P --name tomcat01** --net bridge** tomcat
# 我们可以自定义一个网络!
# --driver bridge 默认桥接模式
# --subnet 192.168.0.0/16 子网
# --gateway 192.168.0.1 网关
[root@localhost ~]# docker network create --driver bridge --subnet 192.168.0.0/16 --gateway 192.168.0.1 mynet
自己的网络建立好了:
测试:
[root@localhost ~]# docker run -d -P --name tomcat-net-01 --net mynet tomcat
[root@localhost ~]# docker run -d -P --name tomcat-net-02 --net mynet tomcat
[root@localhost ~]# docker exec -it tomcat-net-01 ping tomcat-net-02
可以ping的通
============================================================================
4.3 网络连通
当前
#测试 tomcat01(docke0网卡) —> mynet(自定义网卡mynet)
#连通之后就是将tomcat01 放到了mynet网络下?
#一个容器2个ip地址 【阿里云 公网ip 私网ip】
[root@localhost ~]# docker network connect mynet tomcat01
然后,查询下
[root@localhost ~]# docker network inspect mynet
执行[root@localhost ~]# docker inspect tomcat01
测试通不通:[root@localhost ~]# docker exec -it tomcat01 ping tomcat-net-01
[root@localhost redis_my_net]# docker exec -it tomcat-net-01 ping tomcat01