Docker学习笔记2
容器数据卷 什么是容器数据卷 将应用和环境打包成一个镜像!
数据?如果数据都在容器中,那么我们容器删除,数据就会丢失!需求:数据可以持久化
MySQL,容器删除了,删库跑路!需求:MySQL数据可以存储在本地!
容器之间可以有一个数据共享的技术!Docker容器中产生的数据,同步到本地!
这就是卷技术!目录的挂载,将我们容器内的目录,挂载到Linux上面!
总结一句话:容器的持久化和同步操作!容器间也是可以数据共享的!
使用数据卷
方式一 :直接使用命令挂载 -v
1 2 3 4 5 6 7 -v, --volume list Bind mount a volume docker run -it -v 主机目录:容器内目录 -p 主机端口:容器内端口 [root@iz2zeak7 home] [root@iz2zeak7sgj6i7hrb2g862z home]
测试文件的同步
再来测试!
1、停止容器
2、宿主机修改文件
3、启动容器
4、容器内的数据依旧是同步的
好处:我们以后修改只需要在本地修改即可,容器内会自动同步!
实战:安装MySQL 思考:MySQL的数据持久化的问题
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 [root@iz2zeak7sgj6i7hrb2g862z home] docker run --name some-mysql -e MYSQL_ROOT_PASSWORD=my-secret-pw -d mysql:tag -d 后台运行 -p 端口映射 -v 卷挂载 -e 环境配置 -- name 容器名字 $ docker run -d -p 3310:3306 -v /home/mysql/conf:/etc/mysql/conf.d -v /home/mysql/data:/var/lib/mysql -e MYSQL_ROOT_PASSWORD=123456 --name mysql03 mysql:5.7
测试连接 :注意3310端口要在阿里云服务器的安全组中打开,否则无法连接。
当我们在本地用SQLyog新建名称为test的数据库时候,容器容器也会创建
假设我们将包含mysql的容器删除时,
发现,我们挂载到本地的数据卷依旧没有丢失,这就实现了容器数据持久化功能 。
具名和匿名挂载 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 -v 容器内路径! $ docker run -d -P --name nginx01 -v /etc/nginx nginx $ docker volume ls DRIVER VOLUME NAME local 21159 a8518abd468728cdbe8594a75b204a10c26be6c36090cde1ee88965f0d0 local b17f52d38f528893dd5720899f555caf22b31bf50b0680e7c6d5431dbda2802c $ docker run -d -P --name nginx02 -v juming-nginx:/etc/nginx nginx 9663 cfcb1e5a9a1548867481bfddab9fd7824a6dc4c778bf438a040fe891f0ee$ docker volume ls DRIVER VOLUME NAME local 21159 a8518abd468728cdbe8594a75b204a10c26be6c36090cde1ee88965f0d0 local b17f52d38f528893dd5720899f555caf22b31bf50b0680e7c6d5431dbda2802c local juming-nginx $ docker volume inspect juming-nginx [ { "CreatedAt" : "2020-05-23T13:55:34+08:00" , "Driver" : "local" , "Labels" : null, "Mountpoint" : "/var/lib/docker/volumes/juming-nginx/_data" , "Name" : "juming-nginx" , "Options" : null, "Scope" : "local" } ]
所有的docker容器内的卷,没有指定目录的情况下都是在**/var/lib/docker/volumes/自定义的卷名/_data**下,如果指定了目录,docker volume ls 是查看不到的 。
区分三种挂载方式
1 2 3 4 -v 容器内路径 -v 卷名:容器内路径 -v /宿主机路径:容器内路径
拓展:
1 2 3 4 5 6 ro rw $ docker run -d -P --name nginx05 -v juming:/etc/nginx:ro nginx $ docker run -d -P --name nginx05 -v juming:/etc/nginx:rw nginx
初始Dockerfile Dockerfile 就是用来构建docker镜像的构建文件 !命令脚本!先体验一下!
通过这个脚本可以生成镜像 ,镜像是一层一层的,脚本是一个个的命令,每个命令都是一层!
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 $ vim dockerfile1 FROM centos VOLUME ["volume01" ,"volume02" ] CMD echo "-----end-----" CMD /bin/bash -f dockerfile1 -t caoshipeng/centos . $ docker build -f dockerfile1 -t caoshipeng/centos . Sending build context to Docker daemon 2.56 kB Step 1 /4 : FROM centos latest: Pulling from library/centos 8 a29a15cefae: Already exists Digest: sha256:fe8d824220415eed5477b63addf40fb06c3b049404242b31982106ac204f6700 Status: Downloaded newer image for centos:latest ---> 470671670 cac Step 2 /4 : VOLUME ["volume01" ,"volume02" ] ---> Running in c18eefc2c233 Removing intermediate container c18eefc2c233 ---> 623 ae1d40fb8 Step 3 /4 : CMD echo "-----end-----" ---> Running in 70 e403669f3c Removing intermediate container 70 e403669f3c ---> 0 eba1989c4e6 Step 4 /4 : CMD /bin/bash ---> Running in 4342 feb3a05b Removing intermediate container 4342 feb3a05b ---> f4a6b0d4d948 Successfully built f4a6b0d4d948 Successfully tagged caoshipeng/centos:latest $ docker images REPOSITORY TAG IMAGE ID CREATED SIZE caoshipeng/centos latest f4a6b0d4d948 About a minute ago 237 MB
启动自己写的容器镜像
1 2 $ docker run -it f4a6b0d4d948 /bin/bash $ ls -l
这个卷和外部一定有一个同步的目录
查看一下卷挂载
1 2 $ docker inspect ca3b45913df5
测试一下刚才的文件是否同步出去了!
这种方式使用的十分多,因为我们通常会构建自己的镜像!
假设构建镜像时候没有挂载卷,要手动镜像挂载 -v 卷名:容器内路径!
数据卷容器 多个MySQL同步数据 !
命名的容器挂载数据卷!
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 $ docker run -it --name docker01 caoshipeng/centos:latest $ ls bin home lost+found opt run sys var dev lib media proc sbin tmp volume01 etc lib64 mnt root srv usr volume02 CTRL + Q + P $ docker run -it --name docker02 --volumes-from docker01 caoshipeng/centos:latest $ ls bin home lost+found opt run sys var dev lib media proc sbin tmp volume01 etc lib64 mnt root srv usr volume02
1 2 3 4 5 6 7 $ docker run -it --name docker03 --volumes-from docker01 caoshipeng/centos:latest $ cd volume01 $ ls docker01.txt
多个mysql实现数据共享
1 2 3 $ docker run -d -p 3306:3306 -v /home/mysql/conf:/etc/mysql/conf.d -v /home/mysql/data:/var/lib/mysql -e MYSQL_ROOT_PASSWORD=123456 --name mysql01 mysql:5.7 $ docker run -d -p 3310:3306 -e MYSQL_ROOT_PASSWORD=123456 --name mysql02 --volumes-from mysql01 mysql:5.7
结论:
容器之间的配置信息的传递,数据卷容器的生命周期一直持续到没有容器使用为止 。
但是一旦你持久化到了本地,这个时候,本地的数据是不会删除的 !
DockerFile DockerFile介绍 dockerfile
是用来构建docker镜像的文件!命令参数脚本!
构建步骤:
1、 编写一个dockerfile文件
2、 docker build 构建称为一个镜像
3、 docker run运行镜像
4、 docker push发布镜像(DockerHub 、阿里云仓库)
点击后跳到一个Dockerfile
很多官方镜像都是基础包,很多功能没有,我们通常会自己搭建自己的镜像!
官方既然可以制作镜像,那我们也可以!
DockerFile构建过程 基础知识 :
1、每个保留关键字(指令)都是必须是大写字母
2、执行从上到下顺序
3、#表示注释
4、每一个指令都会创建提交一个新的镜像曾,并提交!
Dockerfile是面向开发的,我们以后要发布项目,做镜像,就需要编写dockerfile文件,这个文件十分简单!
Docker镜像逐渐成企业交付的标准,必须要掌握!
DockerFile:构建文件,定义了一切的步骤,源代码
DockerImages:通过DockerFile构建生成的镜像,最终发布和运行产品。
Docker容器:容器就是镜像运行起来提供服务。
DockerFile的指令 1 2 3 4 5 6 7 8 9 10 11 12 FROM MAINTAINER RUN ADD WORKDIR VOLUME EXPOSE CMD ENTRYPOINT ONBUILD COPY ENV
实战测试 scratch 镜像
1 2 3 4 5 6 7 8 9 10 11 12 13 FROM scratchADD centos-7-x86_64-docker.tar.xz / LABEL \ org.label-schema.schema-version="1.0" \ org.label-schema.name="CentOS Base Image" \ org.label-schema.vendor="CentOS" \ org.label-schema.license="GPLv2" \ org.label-schema.build-date="20200504" \ org.opencontainers.image.title="CentOS Base Image" \ org.opencontainers.image.vendor="CentOS" \ org.opencontainers.image.licenses="GPL-2.0-only" \ org.opencontainers.image.created="2020-05-04 00:00:00+01:00" CMD ["/bin/bash" ]
Docker Hub 中 99%的镜像都是从这个基础镜像过来的 FROM scratch ,然后配置需要的软件和配置来进行构建。
创建一个自己的centos
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 $ mkdir dockerfile $ vim mydockerfile-centos FROM centos MAINTAINER cao<1165680007 @qq.com> ENV MYPATH /usr/local WORKDIR $MYPATH RUN yum -y install vim RUN yum -y install net-tools EXPOSE 80 CMD echo $MYPATH CMD echo "-----end----" CMD /bin/bash $ docker build -f mydockerfile-centos -t mycentos:0.1 .
1 2 3 4 5 6 7 8 9 10 11 $ docker images REPOSITORY TAG IMAGE ID CREATED SIZE mycentos 0.1 cbf5110a646d 2 minutes ago 311 MB $ docker run -it mycentos:0.1 $ pwd /usr/local $ vim $ ifconfig $ docker history 镜像id
我们可以列出本地进行的变更历史
我们平时拿到一个镜像,可以用 “docker history 镜像id” 研究一下是什么做的
CMD 和 ENTRYPOINT区别
测试cmd
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 $ vim dockerfile-test-cmd FROM centos CMD ["ls" ,"-a" ] $ docker build -f dockerfile-test-cmd -t cmd-test:0.1 . $ docker run cmd-test:0.1 . .. .dockerenv bin dev etc home $ docker run cmd-test:0.1 -l docker: Error response from daemon: OCI runtime create failed: container_linux.go:349 : starting container process caused "exec: \"-l\": executable file not found in $PATH" : unknown.ERRO[0000 ] error waiting for container: context canceled
测试ENTRYPOINT
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 $ vim dockerfile-test-entrypoint FROM centos ENTRYPOINT ["ls" ,"-a" ] $ docker build -f dockerfile-test-entrypoint -t cmd-test:0.1 . $ docker run entrypoint-test:0.1 . .. .dockerenv bin dev etc home lib lib64 lost+found ... $ docker run entrypoint-test:0.1 -l total 56 drwxr-xr-x 1 root root 4096 May 16 06 :32 . drwxr-xr-x 1 root root 4096 May 16 06 :32 .. -rwxr-xr-x 1 root root 0 May 16 06 :32 .dockerenv lrwxrwxrwx 1 root root 7 May 11 2019 bin -> usr/bin drwxr-xr-x 5 root root 340 May 16 06 :32 dev drwxr-xr-x 1 root root 4096 May 16 06 :32 etc drwxr-xr-x 2 root root 4096 May 11 2019 home lrwxrwxrwx 1 root root 7 May 11 2019 lib -> usr/lib lrwxrwxrwx 1 root root 9 May 11 2019 lib64 -> usr/lib64 ....
Dockerfile中很多命令都十分的相似,我们需要了解它们的区别,我们最好的学习就是对比他们然后测试效果!
实战:Tomcat镜像 1、准备镜像文件 1 准备tomcat 和 jdk 到当前目录,编写好README
2、编写dokerfile 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 $ vim dockerfile FROM centos MAINTAINER cao<1165680007 @qq.com> COPY README /usr/local /README ADD jdk-8u231-linux-x64.tar.gz /usr/local / ADD apache-tomcat-9.0.35.tar.gz /usr/local / RUN yum -y install vim ENV MYPATH /usr/local WORKDIR $MYPATH ENV JAVA_HOME /usr/local/jdk1.8.0 _231 ENV CLASSPATH $JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jarENV CATALINA_HOME /usr/local/apache-tomcat-9.0 .35 ENV CATALINA_BASH /usr/local/apache-tomcat-9.0 .35 ENV PATH $PATH:$JAVA_HOME/bin:$CATALINA_HOME/lib:$CATALINA_HOME/bin EXPOSE 8080 CMD /usr/local /apache-tomcat-9.0.35/bin/startup.sh && tail -F /usr/local /apache-tomcat-9.0.35/logs/catalina.out
3、构建镜像 1 2 $ docker build -t mytomcat:0.1 .
4、run镜像 1 2 3 4 $ docker run -d -p 8080:8080 --name tomcat01 -v /home/kuangshen/build/tomcat/test:/usr/local/apache-tomcat-9.0 .35 /webapps/test -v /home/kuangshen/build/tomcat/tomcatlogs/:/usr/local/apache-tomcat-9.0 .35 /logs mytomcat:0.1
5、访问测试 1 2 $ docker exec -it 自定义容器的id /bin/bash $ cul localhost:8080
6、发布项目 (由于做了卷挂载,我们直接在本地编写项目就可以发布了!)
发现:项目部署成功,可以直接访问!
我们以后开发的步骤:需要掌握Dockerfile的编写!我们之后的一切都是使用docker镜像来发布运行!
发布自己的镜像
发布到 Docker Hub
1、地址 https://hub.docker.com/
2、确定这个账号可以登录
3、登录
1 2 3 4 5 6 7 8 9 $ docker login --help Usage: docker login [OPTIONS] [SERVER] Log in to a Docker registry. If no server is specified, the default is defined by the daemon. Options: -p, --password string Password --password-stdin Take the password from stdin -u, --username string Username $ docker login -u 你的用户名 -p 你的密码
4、提交 push镜像
1 2 3 4 5 6 7 $ docker build -t kuangshen/mytomcat:0.1 . $ docker tag 容器id kuangshen/mytomcat:1.0 $ docker push kuangshen/mytomcat:1.0
发布到 阿里云镜像服务上
看官网 很详细https://cr.console.aliyun.com/repository/
1 2 3 4 5 6 $ sudo docker login --username=zchengx registry.cn-shenzhen.aliyuncs.com $ sudo docker tag [ImageId] registry.cn-shenzhen.aliyuncs.com/dsadxzc/cheng:[镜像版本号] sudo docker tag a5ef1f32aaae registry.cn-shenzhen.aliyuncs.com/dsadxzc/cheng:1.0 $ sudo docker push registry.cn-shenzhen.aliyuncs.com/dsadxzc/cheng:[镜像版本号]
小结
Docker 网络 理解Docker 0 学习之前清空下前面的docker 镜像、容器
1 2 3 4 $ docker rm -f $(docker ps -aq) $ docker rmi -f $(docker images -aq)
测试
三个网络
问题: docker 是如果处理容器网络访问的?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 $ docker run -d -P --name tomcat01 tomcat $ docker exec -it 容器id ip addr $ ip addr 1 : lo: <LOOPBACK,UP,LOWER_UP> mtu 65536 qdisc noqueue state UNKNOWN group default qlen 1 link/loopback 00 :00 :00 :00 :00 :00 brd 00 :00 :00 :00 :00 :00 inet 127.0 .0.1 /8 scope host lo valid_lft forever preferred_lft forever 261 : eth0@if91: <BROADCAST,MULTICAST,UP,LOWER_UP> mtu 1500 qdisc noqueue state UP group default link/ether 02 :42 :ac:12 :00 :02 brd ff:ff:ff:ff:ff:ff link-netnsid 0 inet 172.18 .0.2 /16 brd 172.18 .255.255 scope global eth0 valid_lft forever preferred_lft forever $ ping 172.18 .0.2 PING 172.18 .0.2 (172.18 .0.2 ) 56 (84 ) bytes of data. 64 bytes from 172.18 .0.2 : icmp_seq=1 ttl=64 time=0.069 ms64 bytes from 172.18 .0.2 : icmp_seq=2 ttl=64 time=0.074 ms
原理
1、我们每启动一个docker容器,docker就会给docker容器分配一个ip,我们只要按照了docker,就会有一个docker0桥接模式,使用的技术是veth-pair技术!
https://www.cnblogs.com/bakari/p/10613710.html
再次测试 ip addr
2 、再启动一个容器测试,发现又多了一对网络
3、我们来测试下tomcat01和tomcat02是否可以ping通
1 2 3 4 5 6 7 8 9 10 11 12 $ docker-tomcat docker exec -it tomcat01 ip addr 550 : eth0@if551: <BROADCAST,MULTICAST,UP,LOWER_UP> mtu 1500 qdisc noqueue state UP group default link/ether 02 :42 :ac:11 :00 :02 brd ff:ff:ff:ff:ff:ff link-netnsid 0 inet 172.17 .0.2 /16 brd 172.17 .255.255 scope global eth0 valid_lft forever preferred_lft forever $ docker-tomcat docker exec -it tomcat02 ping 172.17 .0.2 PING 172.17 .0.2 (172.17 .0.2 ) 56 (84 ) bytes of data. 64 bytes from 172.17 .0.2 : icmp_seq=1 ttl=64 time=0.098 ms64 bytes from 172.17 .0.2 : icmp_seq=2 ttl=64 time=0.071 ms
网络模型图
结论:tomcat01和tomcat02公用一个路由器,docker0。
所有的容器不指定网络的情况下,都是docker0路由的,docker会给我们的容器分配一个默认的可用ip。
小结
Docker使用的是Linux的桥接,宿主机是一个Docker容器的网桥 docker0
Docker中所有网络接口都是虚拟的,虚拟的转发效率高(内网传递文件)
只要容器删除,对应的网桥一对就没了!
思考一个场景:我们编写了一个微服务,database url=ip: 项目不重启,数据ip换了,我们希望可以处理这个问题,可以通过名字来进行访问容器 ?
–-link 1 2 3 4 5 6 7 8 9 10 11 12 13 $ docker exec -it tomcat02 ping tomca01 ping: tomca01: Name or service not known $ docker run -d -P --name tomcat03 --link tomcat02 tomcat 5 f9331566980a9e92bc54681caaac14e9fc993f14ad13d98534026c08c0a9aef$ docker exec -it tomcat03 ping tomcat02 PING tomcat02 (172.17 .0.3 ) 56 (84 ) bytes of data. 64 bytes from tomcat02 (172.17 .0.3 ): icmp_seq=1 ttl=64 time=0.115 ms64 bytes from tomcat02 (172.17 .0.3 ): icmp_seq=2 ttl=64 time=0.080 ms
探究:
docker network inspect 网络id 网段相同
docker inspect tomcat03
查看tomcat03里面的/etc/hosts发现有tomcat02的配置
–link 本质就是在hosts配置中添加映射
现在使用Docker已经不建议使用–link了!
自定义网络,不适用docker0!
docker0问题:不支持容器名连接访问!
自定义网络 1 2 3 4 5 6 7 8 docker network connect -- Connect a container to a network create -- Creates a new network with a name specified by the disconnect -- Disconnects a container from a network inspect -- Displays detailed information on a network ls -- Lists all the networks created by the user prune -- Remove all unused networks rm -- Deletes one or more networks
查看所有的docker网络
网络模式
bridge :桥接 docker(默认,自己创建也是用bridge模式)
none :不配置网络,一般不用
host :和所主机共享网络
container :容器网络连通(用得少!局限很大)
测试
1 2 3 4 5 6 7 $ docker run -d -P --name tomcat01 tomcat 等价于 => docker run -d -P --name tomcat01 --net bridge tomcat $ docker network create --driver bridge --subnet 192.168 .0.0 /16 --gateway 192.168 .0.1 mynet
1 $ docker network inspect mynet;
启动两个tomcat,再次查看网络情况
在自定义的网络下,服务可以互相ping通,不用使用–link
我们自定义的网络docker当我们维护好了对应的关系,推荐我们平时这样使用网络!
好处:
redis -不同的集群使用不同的网络,保证集群是安全和健康的
mysql-不同的集群使用不同的网络,保证集群是安全和健康的
网络连通
1 2 3 4 $ docker run -d -P --name tomcat01 tomcat $ docker run -d -P --name tomcat02 tomcat
结论:假设要跨网络操作别人,就需要使用docker network connect 连通!
实战:部署Redis集群
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 docker network create redis --subnet 172.38 .0.0 /16 for port in $(seq 1 6 );\ do \ mkdir -p /mydata/redis/node-${port}/conf touch /mydata/redis/node-${port}/conf/redis.conf cat << EOF >> /mydata/redis/node-${port}/conf/redis.conf port 6379 bind 0.0 .0.0 cluster-enabled yes cluster-config-file nodes.conf cluster-node-timeout 5000 cluster-announce-ip 172.38 .0.1 ${port} cluster-announce-port 6379 cluster-announce-bus-port 16379 appendonly yes EOF done for port in $(seq 1 6 );\ docker run -p 637${port} :6379 -p 1667${port} :16379 --name redis-${port} \ -v /mydata/redis/node-${port} /data:/data \ -v /mydata/redis/node-${port} /conf/redis.conf:/etc/redis/redis.conf \ -d --net redis --ip 172.38.0.1${port} redis:5.0.9-alpine3.11 redis-server /etc/redis/redis.conf docker exec -it redis-1 /bin/sh redis-cli --cluster create 172.38 .0.11 :6379 172.38 .0.12 :6379 172.38 .0.13 :6379 172.38 .0.14 :6379 172.38 .0.15 :6379 172.38 .0.16 :6379 --cluster-replicas 1
docker搭建redis集群完成!
我们使用docker之后,所有的技术都会慢慢变得简单起来!
SpringBoot微服务打包Docker镜像 1、构建SpringBoot项目
2、打包运行
3、编写dockerfile
1 2 3 4 5 FROM java:8 COPY *.jar /app.jar CMD ["--server.port=8080" ] EXPOSE 8080 ENTRYPOINT ["java" ,"-jar" ,"app.jar" ]
4、构建镜像
1 2 3 $ docker build -t xxxxx:xx .
5、发布运行
以后我们使用了Docker之后,给别人交付就是一个镜像即可!