1. Manage data in Docker
默认情况下,所有在容器内部创建的文件被存储在一个可写的容器层。这就意味着:
当容器不存在的时候,数据不能被持久化,而且在容器外部想要读取这些数据十分困难。
容器的可写的层与运行容器的主机密切相关。你不能轻易地移动数据。
要想把数据写到容器的可写层,需要一个存储驱动来管理文件系统。存储驱动提供一个文件系统供Linux内核调用。
Docker有两种方式将容器中的文件存储到主机上,这样的话即使容器停止了,文件依然被持久化下来。这两种方式分别是:volumes和bind mounts。如果你是在Linux上运行Docker,那么你还可以使用tmpfs mount。
2. 选择正确的挂载类型
无论你选择哪种挂载类型,数据看起来是一样的。它以目录或者容器的文件系统中的单个文件的形式暴露出来。
下面这幅图可以帮助我们更好的理解 volumes 、 bind mounts 、 tmpfs mounts 三者的区别
Volumes 存储在由Docker管理的主机文件系统的一部分中(在Linux中是 /var/lib/docker/volumes/ )。非Docker进程不应该去修改这部分文件系统。Volumes是在Docker上持久化数据最好的方式。
Bind mounts 数据可以被储存在主机的任何地方。它们甚至可能是重要的系统文件或目录。Docker主机或Docker容器上的非Docker进程可以随时修改它们。
tmpfs mounts 数据只会被存储到主机的内存中,而且从来不会写到文件系统上。
3. 关于挂载类型
Volumes
Volumes由Docker创建并管理。你可以用docker volume create命令显式地创建一个volume,或者在容器或服务创建的时候创建一个volume。
当你创建了一个volume以后,它被存储到Docker主机上的一个目录下。当你挂载这个volume到一个容器的时候,这个目录就是挂载到容器中的目录。这种方式跟bind mounts很像,除了volumes被Docker管理并与主机的核心功能隔离以外。
一个volume可以同时挂载到多个容器中。当没有一个运行中的容器使用这个volume的时候,这个volume仍然是可用的,并且不会被自动删除。你可以用docker volume prune命令删除未使用的volumes。
当你挂载一个volume的时候,它可能是被命名的或者匿名的。匿名的volumes在它首次被挂载到一个容器中的时候不会被指定一个明确的名字,因此Docker给它们一个随机的名字,以保证它在Docker主机中是唯一的。关于名字,命名的和匿名的volumes在使用上是一样的。
Volumes也支持volume驱动,可以运行你存储数据到远程主机或云上。
Bind mounts
与volumes相比,bind mounts有一些功能限制。当你使用bind mount的时候,主机上的一个文件或目录被挂载到一个容器。这个文件或目录关联主机上的绝对路径。这个文件或目录不需要事先在Docker主机上存在,如果没有则会自动在后台创建。Bind mounts的性能非常好,但它依赖于主机的文件系统上有一个特定的目录结构可用。如果你正在开发一个新的Docker应用,建议用volumes。你不能直接用Docker命令来直接管理bind mounts。
tmpfs mounts
一个tmpfs挂载不会持久化数据到磁盘,无论是在Docker主机上还是在容器中。容器可以在容器的生命周期中使用它来存储非持久性状态或敏感信息。例如,在内部,群集服务使用tmpfs挂载来将秘钥挂载到服务的容器中。
4. volumes最佳实践
Volumes是在Docker容器和服务中保存数据的首选方式。使用volumes的一些场景:
在多个运行的容器之间共享数据。如果你没有显式地创建一个volume,那么在它首次被挂载到一个容器中的时候会被自动创建。当容器停止或被删除以后,这个volume仍然存在。多个容器可以挂载同一个volume,可以是读写或只读的。只有当你显式地删除volumes的时候它们才会被删除。
Docker主机不保证有一个给定的目录或文件结构。Volumes解耦从容器运行时到Docker主机之间的配置。
当你想要存储你的容器的数据到一个远程主机或云上,而不是本地