docker中的导入导出

在 docker 的概念中, 有两套导入导出的概念. 分别是export 和 import还有save 和 load 百度搜索🔍它们的区别, 有一篇被引用了很多次的文章虽然讲的很详细, 还加入了一些实验验证, 但是试验中为了实现同一个效果而使用了 commit 命令更改了镜像内容. 理解和使用起来可能会有些偏差, 本文将以最简单明了的方式介绍 docker 中的两套导入导出功能

export&import save&load
操作对象 容器 镜像
导出对象 tar 文件 tar 文件
导入对象 镜像 镜像
镜像层数 一层 多层

下面我们来一行一行的解释

操作对象

操作的对象不同, export 和 import 是用来导入导出容器用的, 而 save 和 load 是用来导入导出镜像用的.

例如: 你需要把 A 机器上的 容器迁移到 B 机器, 且 容器中有重要的数据需要随之一起迁移的, 就可以使用 export 和 import 参数来导入和导出

例如: 你的服务器不能 pull 下来某个镜像, 但是你有一台海外的机器, 你可以在海外的那台机器上把需要的镜像 pull 下来, 然后把该镜像通过 save 命令导出为一个 tar 包, 再将 tar 包拉回到本地导入

所以, 从功能定位上, 你就记住 save 和 load参数是用来迁移镜像的这个场景就行了, 另外的 export 和 import当然就是用来迁容器的

导出对象

在导出对象这一点, exportsave 导出的文件形式都是一样的, 两者都是导出为一个 tar 包

导入对象

在导入上, importload 通过 tar 包导入的都是一个镜像

导入的镜像层数

最大的区别就在这里, 通过export 和 import导出的容器形成镜像时, 该镜像只有一层

通过saveload 导出的镜像保留了原镜像所有的层次结构, 导出时原镜像有几层, 导入的时候就还是有几层

想导出容器, 但是还想保留层次结构怎么办?

导出容器, 很快就想到唯一一个可以导出容器的工具 export

但是又想保留底层镜像的层次结构, 那么 export 就不符合需求了

想想导出带层次结构的工具就只有镜像导出工具 save 了, 但是容器在镜像层之上还有一层新的数据怎么一起导出去呢?

这个时候就需要引入一个新的参数 commit, 用来保存容器现有的状态为一个新的镜像

比如在 A 机器上运行的 容器是基于 甲方乙方 这个镜像跑起来的, 那么我就可以通过 commit 参数, 将 容器的所有内容保存为一个新的镜像, 名字叫 私人订制 (内含一梗哦😆) 最后我再通过镜像导出工具 save 就可以完整的将 私人订制镜像(也就是 甲容器 )导出为一个 tar 包了

而且包含了 X+1 层镜像, X 层是原镜像 甲方乙方 的所有镜像层数, 1是容器 多的那一层可写层的镜像

总结

总结来说我也希望大家使用 docker 能遵循 docker 的设计初衷, 运行一个无状态的容器. 如果是无状态的容器, 就不会产生 想导出容器, 但是还想保留层次结构怎么办? 这样的奇葩问题了.

所以最后还是要再次强调:

  • export 和 import 是用来导出导入容器用的, 导出的是容器的文件系统, 导出后的镜像只有一层
  • save 和 load 是用来导出导入镜像用的, 导出后保持着原镜像的层次结构

这里我没有详细介绍这四个参数的具体用法, 因为网上已经有很详细的文章介绍啦, 一下就是百度经常能搜到的关于 docker 导入导出的文章

https://my.oschina.net/zjzhai/blog/225112

具体用法可以参考以上文章, 但是概念理解, 我还是觉得自己写的比较清楚😆, 因为我没有故意构造出导出容器和导出镜像结果相同的场景


参考文档

docker export 官方文档https://docs.docker.com/engine/reference/commandline/export/

docker save 官方文档https://docs.docker.com/engine/reference/commandline/save/

docker commit 官方文档https://docs.docker.com/engine/reference/commandline/commit/