目录
目标
前言
概念
官方文档
匿名卷(Anonymous Volumes)
简介
案例
命名卷(Named Volumes)
简介
案例
目标
- 掌握Volume命令
- 通过演示案例,理解数据卷种类与各自的用途。
前言
我们在很多网上教程上可以看到很多老师们往往将数据卷划分为三个种类:匿名卷、命名卷、绑定挂载。但是我去官方网站看了相关文档,文档中十分明确地将绑定挂载(Bind mounts)、数据卷(Volumes)、临时挂载(tmpfs mounts)规定为三个类别。这可能是很多老师从命令的角度去考虑问题,因为-v参数也可以创建绑定挂载。但是对于这些概念性的东西每个人理解方式不同,我们不用过于纠结,只要理解他们之间的区别并且熟练使用就行。相较于绑定挂载,官方推荐优先考虑使用数据卷。
- 其中临时挂载的数据存储在内存中,无法持久化。所以官方将它单独拎出来做一个独立的类型。
- 绑定挂载与数据卷在适用场景上有很大的区别。绑定挂载适用场景偏向于开发和测试阶段,功能更倾向于实时处理,比如代码热部署。
- 数据卷的适用场景偏向生产阶段,功能更偏向于安全和持久化,比如数据的持久化、MQ中间件。
概念
数据卷(Volume)
官方定义
Volumes
Volumes are the preferred mechanism for persisting data generated by and used by Docker containers.
Volumes are often a better choice than persisting data in a container's writable layer, because a volume doesn't increase the size of the containers using it, and the volume's contents exist outside the lifecycle of a given container.
翻译
卷是持久化Docker容器生成和使用数据的首选机制。
卷通常比在容器的可写层中持久化数据更好,因为卷不会增加使用它的容器的大小,而且卷的内容存在于给定容器的生命周期之外。
我的理解
- 将容器内的某些目录和主机内的某些目录做映射处理,主机内的这些目录就叫数据卷。容器内的这些目录中的数据在数据卷中持久化,可以做到删除容器,数据卷中的数据仍然存在。
- 可以把数据卷和MySQL数据库类比:数据卷和MySQL数据库都用于持久化数据。数据卷主要用于跨容器、容器重启、容器重建;MySQL倾向于结构化数据、事务管理、数据一致性等。
官方文档
数据卷的概念及使用方法 | Volumeshttps://docs.docker.com/engine/storage/volumes/
数据卷基本命令 | docker volumehttps://docs.docker.com/reference/cli/docker/volume/
匿名卷(Anonymous Volumes)
简介
- 顾名思义,创建匿名卷时只需要指定容器的目录,不需要关心匿名卷的存储路径。
- 适用于临时数据处理、短期开发任务、重复的测试任务。
案例
创建Nginx容器并把Nginx首页所在目录制作成匿名卷。通过修改容器内的首页内容、修改匿名卷内的首页内容、删除容器观察匿名卷等操作实现对匿名卷的初步了解。
第一步:创建Nginx容器,并将Nginx首页所在的目录做成匿名卷。
#-v表示创建数据卷,-v后面只写了一个目录,则这个目录表示容器内的目录。
docker run -d -p 8848:80 --name nginx -v /usr/share/nginx/html nginx
第二步:查看容器详情。可以发现我们的目标目录(Destination表示容器内的目录)和主机目录(Source)已经产生了关联。由于我们创建的是匿名卷,没有定义卷名,所以Docker为我们自动生成了一串哈希值(Name的值)。
docker inspect 23805528c4af
第三步:去浏览器访问Nginx,页面返回了Nginx的默认首页。此时我们修改容器内Nginx的首页并重启容器,发现首页内容改变了。
localhost:8848
#进入容器
docker exec -it 23805528c4af /bin/bash
#进入Nginx首页目录
cd /usr/share/nginx/html
#修改容器内容。有的镜像非常精简没有vi命令,大家可以用echo创建首页覆盖原来的首页。
echo '<h1>Hello World!<h1>' >index.html
#退出容器。
exit
#关闭容器
docker stop 23805528c4af
#启动容器
docker start 23805528c4af
第四步:根据第二步我们已经看到了数据卷的位置,我们来看看数据卷中的首页内容有没有发生变化。
第五步:修改数据卷中的首页内容,重启容器后访问Nginx,发现页面变化了。
第六步:分别在在容器和主机上,做创建文件、删除文件操作。发现数据卷的内容和容器内对应的目录内容一致。(操作略)
第七步:删除容器,观察数据卷是否还存在。发现还存在。
docker rm nginx
第八步:如果想实现删除容器的同时也删除匿名卷,可以有以下两种方法。
方法一:创建容器的时候使用--rm,这个参数表示容器停止后会自动删除该容器,以及相关的所有资源。
#创建容器
docker run -d -p 8848:80 --name nginx --rm -v /usr/share/nginx/html nginx
#查看数据卷
docker inspect nginx
#停止容器
docker stop nginx
#查看是否还存在数据卷
docker volume ls
方法二:如果创建容器时没有使用--rm,在删除容器时加上-v参数,表示也删除匿名卷。
#创建容器
docker run -d -p 8848:80 --name nginx -v /usr/share/nginx/html nginx
#查看数据卷
docker inspect nginx
#停止容器
docker rm -v nginx
#查看是否还存在数据卷
docker volume ls
命名卷(Named Volumes)
简介
- 创建命名卷时需要定义卷名。从容器的详细来看,命名卷的Name是我们自定义的,匿名卷的Name是Docker自动生成的哈希值。
- 容器的删除不会导致数据卷被删除。虽然匿名卷也能实现,但在创建容器或删除容器时添加特定的参数仍然会使得匿名卷被删除(上面介绍匿名卷的第八步有所体现),而命名卷不会因此被删除。
- 适合做数据持久化,比如数据库的存储目录和日志文件;适合做数据迁移备份;适合多个容器间共享数据。
案例
创建Nginx容器并把Nginx首页所在目录制作成命名卷。通过停止容器、删除容器操作观察命名卷是否仍然存在。
第一步:创建Nginx容器,并将Nginx首页所在的目录做成命名卷。
#卷名是nginx-volume
docker run -d -p 8848:80 --name nginx -rm -v nginx-volume:/usr/share/nginx/html nginx
第二步:停止容器,发现命名卷还是存在。注意:这个容器在创建时用了--rm参数。在之前的案例中,创建匿名卷时使用了这个参数,在停止容器时,匿名卷被删除了,而命名卷保留着。
第三步:创建新容器时不用--rm参数,在删除容器时使用-v参数,看命名卷是否存在。结论是命名卷仍然存在。