当前位置: 首页 > news >正文

镜像与容器

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 事件时并不是二话不说,直接关闭容器,而是分为以下三种情况:

  1. 如果容器中的应用耗尽了主机系统分配给容器的内存限额,就会触发 OOM 事件。此时,此容器将会被强制关闭。但是需要注意的是,此时关闭的容器并不是一个 docker daemon,而是宿主机操作系统。因为一个容器就是一组运行在宿主机操作系统中的进程,宿主机操作系统通过 cgroups 对这组进程设定资源上限,当这些进程申请的资源到达上限时,触发的是宿主机操作系统的内核 OOM 事件,因此最终是由宿主机内核来关闭这些进程。

  2. 如果用户不想关闭这个容器,则可以选择 --oom-kill-disable 来禁用 OOM-Killer。使用这个参数时,仍需要注意,如果使用 -m 设置了此容器内存上限,则当容器到达内存资源上限时,主机不会关闭容器,但也不会继续向此容器继续分配资源,此时容器处于 hung 状态。只需要将最坏的情况封闭在一定范围之内,而不至于蔓延出去。

  3. 如果用户使用了 --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 对象。

相关文章:

  • PostgreSQL 慢 SQL 排查
  • 【MySQL篇】Percona XtraBackup工具备份指南:常用备份命令详解与实践(第二篇,总共五篇)
  • Elasticsearch 批量更新
  • 阿里云国际站:海外视频安全的DRM加密
  • 防溺水预警系统引领水域安全新篇章
  • apache Kylin系列介绍及配置
  • Logback格式简记
  • 026-GeoGebra中级篇-曲线(2)_极坐标曲线、参数化曲面、分段函数曲线、分形曲线、复数平面上的曲线、随机曲线、非线性动力系统的轨迹
  • SpringBoot增加网关服务
  • Linux发行版CentOS 8 利用Docker安装应用
  • 初谈Linux信号-=-信号的产生
  • Kafka Producer发送消息流程之消息异步发送和同步发送
  • 【整体介绍】HTML和JS编写多用户VR应用程序的框架
  • node.js的安装及学习(node/nvm/npm的区别)
  • 【闲聊】-Perl的基础语法
  • 「前端早读君006」移动开发必备:那些玩转H5的小技巧
  • Essential Studio for ASP.NET Web Forms 2017 v2,新增自定义树形网格工具栏
  • Fabric架构演变之路
  • iOS 系统授权开发
  • java小心机(3)| 浅析finalize()
  • leetcode378. Kth Smallest Element in a Sorted Matrix
  • node和express搭建代理服务器(源码)
  • yii2权限控制rbac之rule详细讲解
  • -- 查询加强-- 使用如何where子句进行筛选,% _ like的使用
  • 湖南卫视:中国白领因网络偷菜成当代最寂寞的人?
  • 基于阿里云移动推送的移动应用推送模式最佳实践
  • 前端技术周刊 2019-01-14:客户端存储
  • 使用Envoy 作Sidecar Proxy的微服务模式-4.Prometheus的指标收集
  • 小试R空间处理新库sf
  • Redis4.x新特性 -- 萌萌的MEMORY DOCTOR
  • # Kafka_深入探秘者(2):kafka 生产者
  • #### golang中【堆】的使用及底层 ####
  • ###51单片机学习(2)-----如何通过C语言运用延时函数设计LED流水灯
  • ###STL(标准模板库)
  • #DBA杂记1
  • (1综述)从零开始的嵌入式图像图像处理(PI+QT+OpenCV)实战演练
  • (33)STM32——485实验笔记
  • (7) cmake 编译C++程序(二)
  • (Matlab)使用竞争神经网络实现数据聚类
  • (zt)基于Facebook和Flash平台的应用架构解析
  • (保姆级教程)Mysql中索引、触发器、存储过程、存储函数的概念、作用,以及如何使用索引、存储过程,代码操作演示
  • (超简单)构建高可用网络应用:使用Nginx进行负载均衡与健康检查
  • (二十九)STL map容器(映射)与STL pair容器(值对)
  • (免费领源码)Java#Springboot#mysql农产品销售管理系统47627-计算机毕业设计项目选题推荐
  • (算法)N皇后问题
  • (转)3D模板阴影原理
  • .cfg\.dat\.mak(持续补充)
  • .NET 6 在已知拓扑路径的情况下使用 Dijkstra,A*算法搜索最短路径
  • .net core 客户端缓存、服务器端响应缓存、服务器内存缓存
  • .NET/ASP.NETMVC 深入剖析 Model元数据、HtmlHelper、自定义模板、模板的装饰者模式(二)...
  • .NET6 开发一个检查某些状态持续多长时间的类
  • .NetCore发布到IIS
  • .sys文件乱码_python vscode输出乱码
  • @angular/cli项目构建--Dynamic.Form
  • @EnableConfigurationProperties注解使用