Docker Image (镜像)
Docker 镜像概念
-
Docker iamge 本质上是一个 read-only 只读文件,这个文件包含了文件系统、源码、库文件、依赖、工具等一些运行 application 所必需的文件。
-
可以把 Docker image 理解成一个模板,可以通过这个模板实例化出来很多容器。
-
image 里面是一层层文件系统 Union FS。联合文件系统,可以将几层目录挂在到一起,形成一个虚拟文件系统。
每一层文件系统称为一层 layer,联合文件系统可以对每一层文件系统设置三种权限,只读(readonly)、读写(readwrite)和写出(whiteout-able),但是 docker 镜像中每一层文件系统都是只读的。
构建镜像的时候,从一个最基本的操作系统开始,每个构建的操作都相当于做一层的修改,增加了一层文件系统。一层层网上叠加,上层的修改会覆盖底层该位置的可见性。在使用的时候,只会看到一个完全的整体,不知道里面有几层,也不清楚每一层所做的修改是什么。
镜像主要解决什么问题
三个要点:一致性、压缩、分层存储。
在部署应用时,通常是以手工或者脚本的方式进行。但是这样会导致一个问题,无法让云端和本地的环境一致。而且,用户打包每个应用的过程比较繁琐,需要配置和修改等操作,比较繁琐。
但是,Docker 会把一个镜像制作成一个完整的操作系统所有文件和对应的目录结构,形成一个“压缩包”,并且与本地和测试环境所用的操作系统一样。
docker 最大的贡献就是定义了容器镜像的分层的存储格式。
docker 镜像技术的基础是联合文件系统(UnionFS),其文件系统是分层的,这样既可以充分利用共享层,又可以减少存储空间占用。
docker 镜像提供了一种打包应用程序和预配置服务器环境的便捷方式,可以很方便地将其用于个人用途或与其他 Docker 用户公开共享。
Docker Container (容器)
容器概念
通俗来说,容器是镜像的运行实体。镜像是静态的只读文件,容器是带有运行时需要的可写文件层的,并且容器中的进程属于运行状态。即容器运行着真正的应用进程,它有创建、运行、停止、暂停和删除五种状态。
虽然容器本质上是主机上运行的一个进程,但是容器有自己独立的命名空间隔离和资源限制。也就是说,在容器内部,无法看到主机上的进程、环境变量、网络等信息,这是容器与直接运行在主机上进程的本质区别。
容器是基于镜像创建的可运行实例,并且单独存在,一个镜像可以创建出多个容器。运行容器化环境时,实际上是在容器内部创建该文件系统的读写副本。这将添加一个容器层,该层允许修改镜像的整个版本。
容器的生命周期
各种状态
容器的状态包括:初建状态(created)、运行状态(running)、停止状态(stopped)、暂停状态(paused)、删除状态(deleted)。
各状态之间转换关系如下:
-
docker create:创建容器后,不立即运行,容器进入初建状态。
-
docker run:创建容器,并立即启动进行,进入运行状态。
-
docker start:容器转为运行状态。
-
docker stop:容器将转入停滞状态。
-
docker kill:容器在故障(死机)时,执行 kill(断电),容器转入停止状态,这种操作容器丢失数据,除非必要,否则不建议使用。
-
docker restart:重启容器,容器进入运行状态。
-
docker pause:容器进入暂停状态。
-
docker unpause:取消暂停状态,容器进入运行状态。
-
docker rm:删除容器,容器进行删除状态。
-
killed by out-of-memory(因内存不足被终止):宿主机内存被耗尽,也被称为 OOM,非计划终止,这时需要杀死最吃内存的容器。
-
container process exited(异常终止):出现容器被终止后,将进入 Should restart?(yes需要重启,容器执行 start 命令,转为运行状态;no 不需要重启,容器转为停止状态)。
容器 OOM
Docker 在处理 OOM 事件时并不是二话不说,直接关闭容器,而是分为以下三种情况:
-
如果容器中的应用耗尽了主机系统分配给容器的内存限额,就会触发 OOM 事件。此时,此容器将会被强制关闭。但是需要注意的是,此时关闭的容器并不是一个 docker daemon,而是宿主机操作系统。因为一个容器就是一组运行在宿主机操作系统中的进程,宿主机操作系统通过 cgroups 对这组进程设定资源上限,当这些进程申请的资源到达上限时,触发的是宿主机操作系统的内核 OOM 事件,因此最终是由宿主机内核来关闭这些进程。
-
如果用户不想关闭这个容器,则可以选择 --oom-kill-disable 来禁用 OOM-Killer。使用这个参数时,仍需要注意,如果使用 -m 设置了此容器内存上限,则当容器到达内存资源上限时,主机不会关闭容器,但也不会继续向此容器继续分配资源,此时容器处于 hung 状态。只需要将最坏的情况封闭在一定范围之内,而不至于蔓延出去。
-
如果用户使用了 --oom-kill-disable,但也没有使用 -m 来设定上限,此时此容器会尽可能多地使用主机内存资源。也就是说,主机内存有多大,它就可以用多大。
容器异常退出
每个容器内部都存在一个 Init 进程,容器中其他所有进程都是此进程的子进程。
运行的容器是因为 Init 进程在运行,如果一个子进程因为某种原因造成了退出,那么其父进程也会同步退出,直至 Init 进程也退出。
当 Init 进程退出时,也就代表着此容器被关闭。
Docker 目前没有办法知道此时的进程退出属于正常退出还是异常退出。
当出现容器关闭情况时,Docker Daemon 会尝试再次重新将此容器由 Stopped 状态转为 Running 状态。
只有设置了 --restart 参数的容器,Docker Daemon 才会去尝试启动,否则容器会保持停止状态。
容器暂停
Docker “剥夺”了此容器的 CPU 资源,而其他资源,如 Memory 资源、Network 资源等还保留未动。所以,失去了 CPU 资源的进程,是不会被主机内核系统所调度的,所以此容器就处于“冰封”状态。
常见问题
docker create、docker start 和 docker run 有什么区别?
docker create 命令从 Docker 映像创建一个全新的容器。但是不会立刻运行它。
docker start 命令将启动任何已停止的容器。如果使用 docker create 命令创建容器,则可以使用此命令启动它。
docker run 命令时创建和启动的组合,因为它创建了一个新容器并立即启动它。实际上,如果 docker run 命令在系统上找不到映像,它可以从 Docker Hub 中提取映像。
docker import 和 docker load 有什么区别?
需要先知道 docker save和 docker export 命令:
-
docker save images_name:将一个镜像导出为文件,再使用 docker load 命令将文件导入为一个镜像,会保存该镜像的所有历史记录。比 docker export 命令导出的文件大,因为会保存镜像的所有历史记录。
-
docker export container_id:将一个容器导出为文件,再使用 docker import 命令将容器导入为一个新的镜像,但是相比 docker save 命令,容器文件会丢失原数据和历史记录,仅保存容器当时的状态,相当于虚拟机使用。
既可以使用 docker load 命令来导入镜像库存储文件到本地镜像库,也可以使用 docker import 命令来导入一个容器快照到本地镜像库。
两者的区别在于容器快照将会丢失所有的历史记录和元数据信息,而镜像存储文件将保存完整记录,体积也会更大。
docker rm & docker rmi & docker prune 的差异?
docker rm:删除一个或多个容器。
docker rmi:删除一个或多个镜像。
docker prune:用来删除不再使用的 docker 对象。