玉洪大使 發表於 2019-8-9 22:34:00

你必须知道的Docker数据卷(Volume)

<p>本篇已加入《.NET Core on K8S学习实践系列文章索引》,可以点击查看更多容器化技术相关系列文章。</p>
<h1>一、将Docker数据挂载到容器</h1>
<p>  在Docker中,要想实现数据的持久化(所谓Docker的数据持久化即<em><strong>数据不随着Container的结束而结束</strong></em>),需要将数据从宿主机挂载到容器中。目前Docker提供了三种不同的方式将数据从宿主机挂载到容器中:</p>
<p>  (1)volumes:Docker管理宿主机文件系统的一部分,默认位于 /var/lib/docker/volumes 目录中;(<strong>最常用的方式</strong>)</p>
<p>  <img style="width: 70%; border: 2px solid rgba(221, 221, 221, 1); border-radius: 5px" src="https://img2018.cnblogs.com/blog/381412/201908/381412-20190808225451807-827316452.png" alt=""></p>
<p>  由上图可以知道,目前所有Container的数据都保存在了这个目录下边,由于没有在创建时指定卷,所以Docker帮我们默认创建许多匿名(就上面这一堆很长ID的名字)卷。</p>
<p>  (2)bind mounts:意为着可以存储在宿主机系统的任意位置;(<strong>比较常用的方式</strong>)</p>
<p>  但是,bind mount在不同的宿主机系统时不可移植的,比如Windows和Linux的目录结构是不一样的,bind mount所指向的host目录也不能一样。这也是为什么bind mount不能出现在Dockerfile中的原因,因为这样Dockerfile就不可移植了。</p>
<p>  (3)tmpfs:挂载存储在宿主机系统的内存中,而不会写入宿主机的文件系统;(<strong>一般都不会用的方式</strong>)</p>
<p>  三种方式的示意图如下所示:</p>
<p>  <img style="width: 50%" src="https://img2018.cnblogs.com/blog/381412/201908/381412-20190808225945967-659254433.png" alt=""></p>
<h1>二、Volume的基本使用</h1>
<h2>2.1 管理卷</h2>
<div class="cnblogs_code">
<pre># docker volume create edc-nginx-vol <span style="color: rgba(0, 128, 0, 1)">//</span><span style="color: rgba(0, 128, 0, 1)"> 创建一个自定义容器卷</span>
# docker volume <span style="color: rgba(0, 0, 255, 1)">ls</span> <span style="color: rgba(0, 128, 0, 1)">//</span><span style="color: rgba(0, 128, 0, 1)"> 查看所有容器卷</span>
# docker volume inspect edc-nginx-vol <span style="color: rgba(0, 128, 0, 1)">//</span><span style="color: rgba(0, 128, 0, 1)"> 查看指定容器卷详情信息</span></pre>
</div>
<p>  例如,这里我们创建一个自定义的容器卷,名为"edc-nginx-vol":</p>
<p>  <img style="width: 70%; border: 2px solid rgba(221, 221, 221, 1); border-radius: 5px" src="https://img2018.cnblogs.com/blog/381412/201908/381412-20190808231052990-2110905793.png" alt=""></p>
<h2>2.2 创建使用指定卷的容器</h2>
<p>  有了自定义容器卷,我们可以创建一个使用这个数据卷的容器,这里我们以nginx为例:</p>
<div class="cnblogs_code">
<pre># docker run -d -it --name=edc-nginx -p 8800:80 -v edc-nginx-vol:/usr/share/nginx/<span style="color: rgba(0, 0, 0, 1)">html nginx<br></span></pre>
</div>
<p>  其中,-v代表挂载数据卷,这里使用自定数据卷edc-nginx-vol,并且将数据卷挂载到 /usr/share/nginx/html (这个目录是yum安装nginx的默认网页目录)。</p>
<p>  如果没有通过-v指定,那么Docker会默认帮我们创建匿名数据卷进行映射和挂载。</p>
<p>  创建好容器之后,我们可以进入容器里面看看:</p>
<p>  <img style="width: 70%; border: 2px solid rgba(221, 221, 221, 1); border-radius: 5px" src="https://img2018.cnblogs.com/blog/381412/201908/381412-20190809214509341-1954116672.png" alt=""></p>
<p>&nbsp;  可以看到有两个默认页,这时我们新启动一个SSH连接到宿主机去到刚刚创建的数据卷里边看看:</p>
<p>  <img style="width: 70%; border: 2px solid rgba(221, 221, 221, 1); border-radius: 5px" src="https://img2018.cnblogs.com/blog/381412/201908/381412-20190809214642233-1309841993.png" alt=""></p>
<p>&nbsp;  可以看到,我们可以访问到容器里面的两个默认页面,由此可知,volume帮我们做的类似于一个软链接的功能。在容器里边的改动,我们可以在宿主机里感知,而在宿主机里面的改动,在容器里边可以感知到。</p>
<p>  这时,如果我们手动stop并且remove当前nginx容器,我们会发现容器卷里面的文件还在,并没有被删除掉。</p>
<p>  <img style="width: 70%; border: 2px solid rgba(221, 221, 221, 1); border-radius: 5px" src="https://img2018.cnblogs.com/blog/381412/201908/381412-20190809215104703-539582054.png" alt=""></p>
<p>&nbsp;  由此可以验证,在数据卷里边的东西是可以持久化的。如果下次还需要创建一个nginx容器,那么还是<strong>复用</strong>当前数据卷里面的文件。</p>
<p>  此外,我们还可以启动多个nginx容器实例,并且共享同一个数据卷,复用性和扩展性较强。</p>
<h2>2.3 清理卷</h2>
<p>  如果不再使用自定义数据卷了,那么可以手动清理掉:</p>
<div class="cnblogs_code">
<pre># docker stop edc-<span style="color: rgba(0, 0, 0, 1)">nginx <span style="color: rgba(51, 153, 102, 1)">// 暂停容器实例</span>
# docker </span><span style="color: rgba(0, 0, 255, 1)">rm</span> edc-<span style="color: rgba(0, 0, 0, 1)">nginx <span style="color: rgba(51, 153, 102, 1)">// 移除容器实例</span>
# docker volume </span><span style="color: rgba(0, 0, 255, 1)">rm</span> edc-nginx-vol <span style="color: rgba(51, 153, 102, 1)">// 删除自定义数据卷</span></pre>
</div>
<h1>三、Bind Mounts的基本使用</h1>
<h2>3.1 使用卷创建一个容器</h2>
<div class="cnblogs_code">
<pre>docker run -d -it --name=edc-nginx -v /app/wwwroot:/usr/share/nginx/html nginx</pre>
</div>
<p>  这里指定了将宿主机上的 /app/wwwroot 目录(如果没有会自动创建)挂载到 /usr/share/nginx/html (这个目录是yum安装nginx的默认网页目录)。</p>
<p>  这时我们再次进入容器内部看看:</p>
<p>  <img style="width: 70%; border: 2px solid rgba(221, 221, 221, 1); border-radius: 5px" src="https://img2018.cnblogs.com/blog/381412/201908/381412-20190809220412635-463463597.png" alt=""></p>
<p>&nbsp;  可以看到,与volumes不同,bind mounts的方式会隐藏掉被挂载目录里面的内容(如果非空的话),这里是/usr/share/nginx/html 目录下的内容被隐藏掉了,因此我们看不到。</p>
<p>  但是,我们可以将宿主机上的文件随时挂载到容器中:</p>
<p>  <em>Step1</em>.新建一个index.html</p>
<p>  <img style="width: 70%; border: 2px solid rgba(221, 221, 221, 1); border-radius: 5px" src="https://img2018.cnblogs.com/blog/381412/201908/381412-20190809221829672-355112964.png" alt=""></p>
<p>  <em>Step2</em>.在容器中查看</p>
<p>  <img style="width: 70%; border: 2px solid rgba(221, 221, 221, 1); border-radius: 5px" src="https://img2018.cnblogs.com/blog/381412/201908/381412-20190809221853801-688990775.png" alt=""></p>
<h2>3.2 验证绑定</h2>
<div class="cnblogs_code">
<pre>docker inspect edc-nginx</pre>
</div>
<p>&nbsp;  通过上述命令可以看到一大波配置,我们要关注的是:</p>
<p>  <img style="width: 70%; border: 2px solid rgba(221, 221, 221, 1); border-radius: 5px" src="https://img2018.cnblogs.com/blog/381412/201908/381412-20190809222205514-1629466713.png" alt=""></p>
<h2>3.3 清理</h2>
<div class="cnblogs_code">
<pre>docker stop edc-<span style="color: rgba(0, 0, 0, 1)">nginx
docker </span><span style="color: rgba(0, 0, 255, 1)">rm</span> edc-nginx</pre>
</div>
<p>  同volumes一样,当我们清理掉容器之后,挂载目录里面的文件仍然还在,不会随着容器的结束而消失,从而实现数据持久化。</p>
<h2>3.4 应用案例</h2>
<p>  在服务治理组件中,服务发现组件是一个最常用的组件之一,Consul是一个流行的服务发现开源项目,Consul推荐我们使用配置文件的方式注册服务信息。因此,我们常常会将填写好服务注册配置文件放在宿主机的一个文件目录下将其挂载到Consul的容器指定目录下,如下所示:</p>
<div class="cnblogs_code">
<pre>docker run -d -p <span style="color: rgba(128, 0, 128, 1)">8500</span>:<span style="color: rgba(128, 0, 128, 1)">8500</span> --restart=always \<br>-v /XiLife/consul/data/server1:/consul/data -v /XiLife/consul/conf/server1:/consul/config \<br>-e CONSUL_BIND_INTERFACE=<span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(128, 0, 0, 1)">eth0</span><span style="color: rgba(128, 0, 0, 1)">'</span> --privileged=<span style="color: rgba(0, 0, 255, 1)">true</span> \<br>--name=consul_server_1 consul:<span style="color: rgba(128, 0, 128, 1)">1.4</span>.<span style="color: rgba(128, 0, 128, 1)">4</span> agent -server -bootstrap-expect=<span style="color: rgba(128, 0, 128, 1)">3</span> -ui -node=consul_server_1 -client=<span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(128, 0, 0, 1)">0.0.0.0</span><span style="color: rgba(128, 0, 0, 1)">'</span> \<br>-data-<span style="color: rgba(0, 0, 255, 1)">dir</span> /consul/data -config-<span style="color: rgba(0, 0, 255, 1)">dir</span> /consul/config -datacenter=xdp_dc;</pre>
</div>
<p>  可以看到,我们通过Bind Mounts的方式将宿主机上的/XiLife/consul/data/server1目录挂载到了容器的/consul/data目录下,还将/XiLife/consul/conf/server1目录挂载到了容器的/consul/config目录下,而容器下的两个目录/consul/data和/consul/config则是我们指定的存放agent数据和配置文件的地方。因此,宿主机上的配置文件的变化会及时反映到容器中,比如我们在宿主机上的目录下更新了配置文件,那么只需要reload一下Consul的容器实例即可:</p>
<div class="cnblogs_code">
<pre>docker exec consul-server consul reload</pre>
</div>
<p>  *.这里的consul-server是容器的名字,consul reload是重新加载的命令(非restart)。</p>
<h1>四、小结</h1>
<p>  本文探索了Docker的数据卷及挂载数据到容器的两种主要方式Volumes和Bind Mounts,并介绍基本的使用方式和步骤,通过数据卷我们可以实现Docker的数据持久化,在实际应用中比较广泛。</p>
<h1>参考资料</h1>
<div>(1)李振良,《Docker Volume详解》</div>
<div>(2)CloudMan,《每天5分钟玩转Docker容器技术》</div>
<div>(3)阿龙,《Docker存储卷详解》</div>
<p>&nbsp;</p>
<div id="Copyright">
<p>作者:周旭龙</p>
<p>出处:https://edisonchou.cnblogs.com</p>
<p>本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接。</p>
</div>

</div>
<div id="MySignature" role="contentinfo">
    <div align="center"><img border="0" src="http://service.t.sina.com.cn/widget/qmd/2068032061/d643d182/10.png"></div><br><br>
来源:https://www.cnblogs.com/edisontalk/p/docker_volumes_introduction.html
頁: [1]
查看完整版本: 你必须知道的Docker数据卷(Volume)