数据共享是volume的关键特性,今天我们来看一下通过volume实现容器与host、容器与容器之间共享数据。

一、容器与host共享数据

在上一篇中介绍到的bind mount和docker manage volume,它们都可以实现容器与host之间共享数据,只是方式有所区别,bind mount在容器启动前便指定了volume所在host数据目录,并挂载到容器中了,容器启动后,我们向此目录写入数据,容器也能使用这些数据,而对于docker manage volume,有所不同,它在容器启动时才能确定volume所在host的目录,所以这里就需要用到docker cp 命令,它可以在容器和host之间复制数据。

$ sudo docker run -d -p 80:80 -v /usr/local/apache2/htdocs httpd
0320b31996408b61a1bc363f999509f3bfdc17ca292dd08bd5f7496edb7c8947
$ sudo docker cp ~/htdocs/index.html 0320b31996:/usr/local/apache2/htdocs
$ curl http://127.0.0.1:80
<h1>update page</h1>

二、容器之间共享数据

1、bind mount

第一种方式还是bind mount,将要共享的数据通过bind mount挂载到多个容器上,例如,我们启动三个httpd容器,并让它们挂载相同的htdocs。

$ sudo docker run --name web1 -d -p 80 -v ~/htdocs:/usr/local/apache2/htdocs httpd
f5a911434445f431d511b5292112fe0f9b9b44b868f98561feba895107e0cb40
$ sudo docker run --name web2 -d -p 80 -v ~/htdocs:/usr/local/apache2/htdocs httpd
61e43e584c33efdab7ef5c1a72a5a4e9a0d40731d860e5fd9fb23de0a7c767df
$ sudo docker run --name web3 -d -p 80 -v ~/htdocs:/usr/local/apache2/htdocs httpd
6494f68481a0f1971fdc33c13b969706c69b8ecaedc39aa707b6d35d32b6fba0
$ sudo docker ps
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
6494f68481a0 httpd "httpd-foreground" 11 seconds ago Up 9 seconds 0.0.0.0:49155->80/tcp, :::49155->80/tcp web3
61e43e584c33 httpd "httpd-foreground" 19 seconds ago Up 17 seconds 0.0.0.0:49154->80/tcp, :::49154->80/tcp web2
f5a911434445 httpd "httpd-foreground" 28 seconds ago Up 27 seconds 0.0.0.0:49153->80/tcp, :::49153->80/tcp web1
$ curl http://127.0.0.1:49155
<h1>hello docker</h1>
$ curl http://127.0.0.1:49154
<h1>hello docker</h1>
$ curl http://127.0.0.1:49153
<h1>hello docker</h1>

2、volume container

volume container是专门为其他容器提供volume的容器,它提供的volume可以是bind mount也可以是docker manage volume,我们首先创建一个volume container:

$ sudo docker create --name vc_data -v ~/htdocs:/usr/local/apache2/htdocs -v /other/useful/tools busybox
178dd66f492dbe0485816a9f6f9ecde97c03e322841e6b38f4a2c7c439c6f020

我们挂载了两个volume,一个是使用bind mount挂载web静态页面,另一个使用docker manage volume挂载常用工具。注意我们使用docker create命令,这是因为volume container的作用只是提供数据,它本身不需要处于运行状态。

通过docker inspect可以看到这两个volume:

$ sudo docker inspect vc_data
[
...
"Mounts": [
{
"Type": "bind",
"Source": "/home/yangye/htdocs",
"Destination": "/usr/local/apache2/htdocs",
"Mode": "",
"RW": true,
"Propagation": "rprivate"
},
{
"Type": "volume",
"Name": "e3ba7240ef18694660ed3da4e6976fa35bb5c2d3c26f91593ea30d26289e0fd6",
"Source": "/var/lib/docker/volumes/e3ba7240ef18694660ed3da4e6976fa35bb5c2d3c26f91593ea30d26289e0fd6/_data",
"Destination": "/other/useful/tools",
"Driver": "local",
"Mode": "",
"RW": true,
"Propagation": ""
}
...
]

接着我们在启动容器时通过--volumes_from使用刚创建的vc_data:

$ sudo docker run --name web1 -d -p 80 --volumes-from vc_data httpd
14131cc7057528ac892b2661ed46e543bb30b65af789d83f0770dab4ebb06a33
$ sudo docker run --name web2 -d -p 80 --volumes-from vc_data httpd
320a11bcb64ee032c65b65ee744a7afd6ebc7a24330e68c688a4b9c4b5e5eb22
$ sudo docker run --name web3 -d -p 80 --volumes-from vc_data httpd
f4c4bed873f3ad76b37136fc93dd915c6e4915897e7628f3b1ca1d5fc30e920a

我们以web1为例看看它的volume是否正确:

$ sudo docker inspect web1
[
...
"Mounts": [
{
"Type": "bind",
"Source": "/home/yangye/htdocs",
"Destination": "/usr/local/apache2/htdocs",
"Mode": "",
"RW": true,
"Propagation": "rprivate"
},
{
"Type": "volume",
"Name": "e3ba7240ef18694660ed3da4e6976fa35bb5c2d3c26f91593ea30d26289e0fd6",
"Source": "/var/lib/docker/volumes/e3ba7240ef18694660ed3da4e6976fa35bb5c2d3c26f91593ea30d26289e0fd6/_data",
"Destination": "/other/useful/tools",
"Driver": "local",
"Mode": "",
"RW": true,
"Propagation": ""
}
...

可见,web1使用的就是vc_data的volume,我再来验证下数据共享的效果:

$ sudo docker ps
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
f4c4bed873f3 httpd "httpd-foreground" 10 minutes ago Up 10 minutes 0.0.0.0:49158->80/tcp, :::49158->80/tcp web3
320a11bcb64e httpd "httpd-foreground" 10 minutes ago Up 10 minutes 0.0.0.0:49157->80/tcp, :::49157->80/tcp web2
14131cc70575 httpd "httpd-foreground" 10 minutes ago Up 10 minutes 0.0.0.0:49156->80/tcp, :::49156->80/tcp web1
$ curl http://127.0.0.1:49158
<h1>hello docker</h1>
$ curl http://127.0.0.1:49157
<h1>hello docker</h1>
$ curl http://127.0.0.1:49156
<h1>hello docker</h1>

三个容器都共享了volume_container中的volume,相比于bind mount,我们发现volume container具备以下特点:

1)不必为每一个容器指定host path,所有path都在volume container中定义好了,容器只需和volume container关联,实现了容器与host的解耦;

2)使用volume container的容器,其mount point都是一致的,有利于配置的规范和标准化,当然这样也存在一定的局限使用时需要综合考虑。

Docker 与 K8S学习笔记(十 二)容器间数据共享的更多相关文章

  1. Docker 与 K8S学习笔记(二十)—— 使用Downward API向容器注入Pod信息

    Kubernetes在创建Pod时,会为Pod和容器设置一些额外的信息,比如Pod名称.Pod IP.Node IP.Label.Annotation.资源限制等,我们经常会在应用程序中使用到这些数据 ...

  2. Docker 与 K8S学习笔记(二)—— 容器核心知识梳理

    本篇主要对容器相关核心知识进行梳理,通过本篇的学习,我们可以对容器相关的概念有一个全面的了解,这样有利于后面的学习. 一.什么是容器? 容器是一种轻量级.可移植.自包含的软件打包技术,使应用程序可以在 ...

  3. Docker 与 K8S学习笔记(二十二)—— 高效使用kubectl的小技巧

    kubectl作为我们主要的操作K8S的工具,其具备非常丰富的功能,但是如果不经过打磨,使用起来还是存在诸多不便,今天我们来看看如何将我们的kubectl打磨的更加易用. 一.命令自动补全 kubec ...

  4. Docker 与 K8S学习笔记(二十五)—— Pod的各种调度策略(上)

    上一篇,我们学习了各种工作负载的使用,工作负载它会自动帮我们完成Pod的调度和部署,但有时我们需要自己定义Pod的调度策略,这个时候该怎么办呢?今天我们就来看一下如何定义Pod调度策略. 一.Node ...

  5. Docker 与 K8S学习笔记(二十四)—— 工作负载的使用

    我们前面讲了很多关于Pod的使用,但是在实际应用中,我们不会去直接创建Pod,我们一般通过Kubernetes提供的工作负载(Deployment.DeamonSet.StatefulSet.Job等 ...

  6. Docker 与 K8S学习笔记(二十三)—— Kubernetes集群搭建

    小伙伴们,好久不见,这几个月实在太忙,所以一直没有更新,今天刚好有空,咱们继续k8s的学习,由于我们后面需要深入学习Pod的调度,所以我们原先使用MiniKube搭建的实验环境就不能满足我们的需求了, ...

  7. python3.4学习笔记(十二) python正则表达式的使用,使用pyspider匹配输出带.html结尾的URL

    python3.4学习笔记(十二) python正则表达式的使用,使用pyspider匹配输出带.html结尾的URL实战例子:使用pyspider匹配输出带.html结尾的URL:@config(a ...

  8. Go语言学习笔记十二: 范围(Range)

    Go语言学习笔记十二: 范围(Range) rang这个关键字主要用来遍历数组,切片,通道或Map.在数组和切片中返回索引值,在Map中返回key. 这个特别像python的方式.不过写法上比较怪异使 ...

  9. Docker 与 K8S学习笔记(九)—— 容器间通信

    容器之间可通过IP.Docker DNS Server或joined三种方式进行通信,今天我们来详细学习一下. 一.IP通信 IP通信很简单,前一篇中已经有所涉及了,只要容器使用相同网络,那么就可以使 ...

随机推荐

  1. 使用CCS10新建TMS320F28335工程并闪烁LED(流水灯)程序

    学习TMS320F28335使用Code Composer Studio 10.4.0下载和安装本文不再叙述. 1. 新建工程 1.1选择目录新建工作区 1.2打开软件界面如下图所示: 1.3选择新建 ...

  2. GaussDB(DWS)中共享消息队列实现的三大功能

    摘要:本文将详细介绍GaussDB(DWS)中共享消息队列的实现. 本文分享自华为云社区<GaussDB(DWS)CBB组件之共享消息队列介绍>,作者:疯狂朔朔. 1)共享消息队列是什么? ...

  3. MySQL 的日志:binlog

    前言:binlog 用于记录数据库执行写入性操作的日志信息,以二进制的形式保留在磁盘中.它是由 Server 层进行记录的,使用任何存储引擎都会产生 binlog. 实验准备 我们通过 Docker ...

  4. PowerShell配置文件后门

      PowerShell 配置文件是在 PowerShell 启动时运行的脚本.   在某些情况下,攻击者可以通过滥用PowerShell配置文件来获得持久性和提升特权.修改这些配置文件,以包括任意命 ...

  5. centos使用docker安装ActiveMQ

    拉取镜像 docker pull webcenter/activemq 启动镜像 docker run --name=activemq -itd -p 8161:8161 -p 61616:61616 ...

  6. linux(centos)系统安装activemq

    activemq是消息中间件,可以用来 解耦.消峰.异步 需要先安装jdk环境:https://www.cnblogs.com/pxblog/p/10512886.html 1.下载文件 (也可以直接 ...

  7. JAVA获取当前年份,月份、日期、小时、分钟、秒等

    import java.util.Calendar; public class Main { public static void main(String[] args) { Calendar cal ...

  8. SpringBoot使用 MyBatis Plus 实现物理分页查询

    一.分页配置在MyBatis Plus 可以直接使用selectPage这样的分页,但返回的数据确实是分页后的数据,但在控制台打印的SQL语句其实并没有真正的物理分页,而是通过缓存来获得全部数据中再进 ...

  9. C语言之字符串替换库函数replace

    头文件 #include <algorithm> 例子 下面的代码, 将字符串中的 /替换为\ std::string str("C:/demo/log/head/send&qu ...

  10. AQS的原理及源码分析

    AQS是什么 AQS= volatile修饰的state变量(同步状态) +FIFO队列(CLH改善版的虚拟双向队列,用于阻塞等待唤醒机制) 队列里维护的Node节点主要包含:等待状态waitStat ...