Docker 容器数据 持久化(系统学习Docker05)

写在前面

本来是可以将数据存储在 容器内部 的。但是存在容器内部,一旦容器被删除掉或者容器毁坏(我亲身经历的痛,当时我们的大数据平台就是运行在docker容器内,有次停电后,不管怎样容器都起不来。以前的同事也没有将数据映射到外面 。搞得最后我们重新导的数据,痛的领悟啊~~)。
再就是如果不将数据映射出来 ,比如要想使用容器内部的web日志,也是比较难搞,反正不就是其它容器/服务不是很方便。

大体上来说,一般只要使用Docker都会将 重要的数据 进行持久化。

本文有任何问题,欢迎留言指正~

Docker 持久化存储技术

默认仅在Container Layer(层) 保存的文件数据,删除容器就没有了

Data Volume(卷):数据卷,数据挂载相关,外挂一个存储


这个外挂的存储 ,即:Volume 有三种类型:

Volume 卷的三种类型:

  1. Volume
    Docker 自己管理的 Data Volume ,由docker后台自动创建,我们不要去 手动 更改映射出来的文件内容。挂载点 存于主机文件系统中的某个区域docker area(linux 一般是存储在 /var/lib/docker/volumes/ 目录下)。

    这种持久化方式是很推荐的*****。

  2. Bind Mounting
    将当前主机目录 与 指定的容器内部目录进行绑定 --> 实现数据共享

    我们在开发的时候比较推荐这种方式,实现数据共享

    Docker的 数据共享 技术 能极大提高开发人员的开发效率

    配合chrome插件super auto refresh plus,边写代码,边看运行结果。相当舒服~。

  3. tmpfs mount(Linux中):存于内存中(注意,并不是持久化到磁盘)。在容器的生命周期中,它能被容器用来存放非持久化的状态或敏感信息

    上图 卷的三种类型,后文主要根据代码来讲解这几种类型

Volume 相关的常用命令:

列出所有卷:docker volume ls

删除某个卷:docker volume rm 卷名
删除所有未使用的卷:docker volume prune # 慎用
创建:docker volume create

查看某个卷的元信息:docker volume inspect 卷名



容器数据 持久化 实操

类型一:Volume
如果没有显式创建卷,一个卷会在最开始 挂载<VOLUME["/var/lib/mysql"] > 时被创建。当容器停止时,卷仍然存在。多个容器可以通过read-write或read-only的方式使用同一个卷。只有在显式删除时,卷才会被删除。

查看官方mysql Dockerfile 中定义的持久化存储方式:
https://github.com/docker-library/mysql/blob/6952c5d5a9889311157362c528d65dc2e37ff660/5.7/Dockerfile

发现是通过 VOLUME["/var/lib/mysql"] 这样的方式将数据持久化到本地的
意思是指,需要将容器内部/var/lib/mysql 目录下的数据 映射出去
不指定其实是不会将数据映射出来的。
映射出来的位置是在/var/lib/docker/volumes/  ,他会自己在该目录下生成一个volume ID
实际的数据就是保存在其中的。

代码验证上面这段话:
docker volume ls
docker run -d --name mysql1 -e MYSQL_ALLOW_EMPTY_PASSWORD=true mysql:5.7
[[email protected] ~]# docker ps
cef34ac36e63        mysql:5.7           "docker-entrypoint.s…"   16 seconds ago      Up 15 seconds       3306/tcp, 33060/tcp                          mysql1
[[email protected] ~]#
docker exec -it mysql1 bash
mysql -u root
create database test1;
quit
exit
回到本地主机:
docker volume ls
docker volume inspect [volume ID]
详细信息中可以看见挂载点
cd 进去,可以看见实际的数据库文件是真实存在的。
/var/lib/docker/volumes/1b7400bf5cca58f06957e865ab7c072b4cd5961c56d536357fb69575985ecbac/_data
[[email protected] _data]# ls
auto.cnf    client-cert.pem  ibdata1      ibtmp1              private_key.pem  server-key.pem
ca-key.pem  client-key.pem   ib_logfile0  mysql               public_key.pem   sys
ca.pem      ib_buffer_pool   ib_logfile1  performance_schema  server-cert.pem  test1
[[email protected] _data]# 

其实这个时候你把容器停止,删除容器。数据库的文件还是会在本地保留的
# 强制删除运行中的容器
docker rm -f mysql1
# 删除volume
docker volume rm VOLUME NAME 

【升级1】
上面的方式有一个不好的地方就是:由于没有指定映射的对应目录,自动为我们生成了一个很长的Volume ID,
这个其实是不太好管理的。最好的方式是,我们自己指定名称。

具体如下:
通过-v 指定映射
docker run -d -v mysql:/var/lib/mysql --name mysql2 -e MYSQL_ALLOW_EMPTY_PASSWORD=true mysql:5.7
[[email protected] ~]# docker volume ls
DRIVER              VOLUME NAME
local               mysql
[[email protected] ~]#
发现是我们自己指定的名称mysql 

【升级2】
上面的还有一个不好的一点 就是不太方便移植映射出来,
我们如果可以将映射出来的目录指定到 Dockerfile 同一级目录下就更好了,
要想实现这个 可以通过编写 docker-compose.yml 文件,在文件中定义volume

volume这种方式可以很好的将数据持久化的本地文件系统中,【推荐】

类型二:Bind Mounting
将当前目录外部 与 指定的容器内部目录进行绑定 --> 实现数据共享

案例中 实现容器内部 /usr/share/nginx/html 与 本机主机 /root/docker-nginx
两个目录下的文件共享

案例:


[[email protected] ~]# mkdir docker-nginx
[[email protected] ~]# cd docker-nginx/

[[email protected] docker-nginx]# vim Dockerfile
FROM nginx:latest
WORKDIR /usr/share/nginx/html
COPY index.html index.html

[[email protected] docker-nginx]# docker build -t liuge36docker/my-nginx .
Sending build context to Docker daemon  3.072kB
Step 1/3 : FROM nginx:latest
 ---> 2073e0bcb60e
Step 2/3 : WORKDIR /usr/share/nginx/html
 ---> Running in 8bdefed78445
Removing intermediate container 8bdefed78445
 ---> 747647be78f1
Step 3/3 : COPY index.html index.html
 ---> b6a5afe9dcf8
Successfully built b6a5afe9dcf8
Successfully tagged liuge36docker/my-nginx:latest
[[email protected] docker-nginx]# 

# 创建容器
将当前目录与 指定的容器内部目录进行绑定
[[email protected] docker-nginx]# docker rm -f web
web
[[email protected] docker-nginx]# docker run -d -v $(pwd):/usr/share/nginx/html -p 8777:80 --name web liuge36docker/my-nginx
4822d9626b36c2b539527a9ba8d9cdebfaee19762b02f957559e4f2ae6e2d281

[[email protected] docker-nginx]# docker exec -it web /bin/bash
[email protected]:/usr/share/nginx/html# ls
Dockerfile  index.html
[email protected]:/usr/share/nginx/html# touch hello.txt
[email protected]:/usr/share/nginx/html# exit
exit

[[email protected] docker-nginx]# ls
Dockerfile  hello.txt  index.html
[[email protected] docker-nginx]# echo  "1111" >> hello.txt
[[email protected] docker-nginx]# docker exec -it web /bin/bash

[email protected]:/usr/share/nginx/html# cat hello.txt
1111
[email protected]:/usr/share/nginx/html#

后面我可能会根据 SpringBoot 单独写一篇文章 ,看看到底这个代码共享 该怎么去做。

本文比较基础,后面我再写一下关于 容错、 备份、恢复、迁移数据卷 相关的东西。

尽情期待~

有什么问题,欢迎留言讨论~~

更多文章:系统学习Docker 不迷路:https://blog.csdn.net/liuge36/category_7651616.html

原文地址:https://www.cnblogs.com/liuge36/p/12297866.html

时间: 2024-08-01 01:37:35

Docker 容器数据 持久化(系统学习Docker05)的相关文章

Docker 容器数据持久化

Docker Volumn 卷 提供了一种容器数据持久化的技术,可以将容器中的目录挂载至本机上.同时也为容器之间的数据共享提供了方便. 不指定挂载到宿主机 比如运行docker run -d --name nginx -v /usr/share/nginx/html whoami/hello-nginx -d 在daemon中运行 --name nginx 将容器命名为nginx -v /usr/share/nginx/html 指定需要持久化的目录,即需要挂载的目录. 使用sudo docke

Docker 数据卷-Docker容器数据持久化方式

以tomcat为例,若是我们需要启动一个docker tomcat 容器,并将我们的工程放进tomcat启动 在没有数据卷的情况下,我们所采用的方式: 利用dockerfile将工程拷贝到tomcat 的 webapps目录下创建一个新的镜像,然后通过这个新的镜像来创建容器并启动,以此部署. 这种方式固然好用,但是却需要创建新的镜像,假如需要部署新的工程,又要创建新的镜像.很麻烦 而且,这种方式会造成重复写的问题,就是我们的工程文件实质上在docker容器和宿主机上写入了两次.造成性能损耗. 还

自己学Docker:5.Docker的数据持久化之数据卷

首先,别忘记之前的两个问题: 1. 如何保存我们在容器里的修改? 2. 如果apt-get如果不能安装时,如何在Docker中安装软件? 删除镜像命令 在此之前,先记一个命令, sudo docker rm ID/NAME 即删除创建的镜像.如 现在删除id为"cc8a23b1d624"的镜像. sudo docker rm cc8a23b1d624 可以看到,id为cc8a23b1d624的镜像被删除了. 而且可以发现,本地的/var/lib/docker/containers目录下

(九) Docker 容器数据卷(挂载)

一.容器数据卷 docker 容器数据卷是容器在运行过程中持久化数据的一种方式,卷是目录或文件,存在于一个或多个容器中,由docker 挂载到容器,但不属于联合文件系统,因此能够绕过Union File System  提供一些用于持续存储或共享数据的特性:卷的设计目的就是数据持久化,完全独立于容器的生命周期,因此docker不会在容器删除时,删除其挂载的数据卷. 特点: 数据卷可在容器之间共享或重用数据. 卷中的更改可以直接生效. 数据卷中的更改,不会包含在镜像中的更新. 数据卷的生命周期一直

5、docker容器数据卷

Docker容器数据卷 是什么 一句话:有点类似我们Redis里面的rdb和aof文件 能干嘛 容器的持久化 容器间继承+共享数据 数据卷 容器内添加 直接命令添加 命令 docker run -it -v /宿主机绝对路径目录:/容器内目录 镜像名 查看数据卷是否挂载成功 容器和宿主机之间数据共享 容器停止退出后,主机修改后数据是否同步 命令(带权限) docker run -it -v /宿主机绝对路径目录:/容器内目录:ro 镜像名 DockerFile添加 根目录下新建mydocker文

数据持久化编程学习总结

一.JDBC编程 1. 使用JDBC规范 在数据库编程方面,最先使用的数据持久化技术无疑是JDBC 可以说JDBC(Java Data Base Connectivity)是学习其它数据持久化技术的基础 Java中访问数据库使用的就是JDBC,基本操作有CRUD(Create-Read-Update-Delete) JDBC定义了数据库的连接,SQL语句的执行以及查询结果集的遍历,一般操作步骤如下: 1.      注册驱动:DriverManager.registerDriver(driver

解决docker容器中Centos7系统的中文乱码

解决docker容器中Centos7系统的中文乱码问题有如下两种方案: 第一种只能临时解决中文乱码: 在命令行中执行如下命令: $>localedef -i zh_CN -f UTF-8 zh_CN.UTF-8 $>yum -y install kde-l10n-Chinese && yum -y reinstall glibc-common $>localedef -c -f UTF-8 -i zh_CN zh_CN.utf8 $>export LC_ALL=zh

docker mysql 数据持久化到本地、设置不区别表名大小写

Docker MySQL 把数据存储在本地目录,很简单,只需要映射本地目录到容器即可 1.加上-v参数 $ docker run -d -e MYSQL_ROOT_PASSWORD=admin --name mysql -v /data/mysql/data:/var/lib/mysql -p 3306:3306 mysql 还可以指定配置文件 docker run -d -e MYSQL_ROOT_PASSWORD=admin --name mysql -v /data/mysql/my.cn

docker 12 docker容器数据卷

数据卷概念 ♣我们知道,当我们把一个运行中的容器关闭后,容器里的数据就没有了(如果你做了docker commit操作,数据会保留到新的镜像里面).所以我们就需要用容器数据卷把容器数据进行持久化储存. ♣还有一种情况,就是希望容器之间能共享数据,这时也需要容器数据卷. ♣一句话,数据卷就是用来解决数据持久化和数据共享的. ♣卷就是目录或者文件,存在一个或者多个容器中,由docker宿主机挂载到容器上,但是不属于联合文件系统,因此能绕过联合文件系统提供一些用于持久化存储或者共享数据的特性. ♣卷的