纯理论容器实现的原理

近期在复习容器的原理,希望这篇文章可以帮助到大家。

一、什么是容器?

        容器本质上就是主机上的一个进程。这个进程拥有自己的用户空间并且和主机共享内核空间。

        容器内的进程可以通过系统调用与内核进行交互,使用内核提供的各种功能和资源。

扩展:什么是内核空间以及用户空间?

        Linux 操作系统分为内核空间(Kernel Space)和用户空间(User Space)。内核空间是由操作系统内核使用的内存区域,用于执行核心功能,如设备驱动、内存管理、进程调度等。用户空间则是给用户程序提供的运行环境,用户程序在用户空间中运行,通过系统调用(system call)与内核空间进行交互。

二、namespace是什么?

        1、Namespace简介

Namespace是Linux基于内核实现的资源视图隔离功能,使得不同Namespace的资源彼此不可见。

        2、容器常见的隔离如下:      

  1. PID Namespace(进程隔离): 每个容器都拥有自己的 PID Namespace,使得容器内的进程只能看到自己所属的 PID 命名空间中的进程,从而实现进程的隔离。

  2. Network Namespace(网络隔离): 每个容器都有自己的 Network Namespace,使得容器内的网络栈与宿主机的网络栈相互隔离,每个容器拥有自己的网络接口、IP 地址、路由表等网络资源,从而实现网络的隔离。

  3. Mount Namespace(文件系统隔离): 每个容器都有自己的 Mount Namespace,使得容器内的文件系统与宿主机的文件系统相互隔离,每个容器拥有自己的根文件系统,可以挂载自己的文件系统,从而实现文件系统的隔离。

  4. UTS Namespace(主机名与域名隔离): 每个容器都有自己的 UTS Namespace,使得容器内的主机名与域名与宿主机相互隔离,每个容器拥有自己的主机名和域名,从而实现主机名与域名的隔离。

    1. 重点:

      上述隔离都可以通过很简单的实验来证明。创建容器执行:ps -aux、查看userid、查看ifconfig、查看自己的目录是否和主机的文件目录内容相同、以及修改主机名是否对主机有影响。

三、怎么理解联合文件系统?

        联合文件系统也是容器镜像的文件系统,打包时每一层均作为可读层(LowerDir),如果修改复制可读层到可写层(UpperDir)镜像修改,可读层不可变,用户看到的是可读和可写层合并后的Rootfs.

在Docker中,联合文件系统被广泛应用于容器的镜像管理中。下面是Docker中涉及到的几个重要概念:

  1. LowerDir:LowerDir是指联合文件系统中底层的只读文件系统层。在Docker中,LowerDir通常指的是镜像的分层文件系统,即容器的基础镜像和其上的其他镜像层。

  2. UpperDir:UpperDir是指联合文件系统中的可写层,用于保存容器的修改或新增的文件。在Docker中,UpperDir存储的是容器运行时产生的文件变化。

  3. MergedDir:MergedDir是LowerDir和UpperDir合并后形成的联合文件系统的可见部分。在MergedDir中,会包含LowerDir和UpperDir中的文件,以及它们的修改和新增。

  4. WorkDir:WorkDir是联合文件系统的工作目录,用于处理文件系统的合并操作。在Docker中,WorkDir通常指的是容器运行时的临时工作目录。

  5. Rootfs:Rootfs是指容器的根文件系统,即容器中的文件系统树的顶层目录。在Docker中,Rootfs是由LowerDir和UpperDir合并而成的,它包含了容器中所有的文件和目录。​​​​​​​

四、什么是Mount namespace?

  1. 简单的说就是将目录的挂载点挂载通过系统调用Clone将新的mnt namespace挂载到新的namespace中,来隔离挂点。​​​​​​​

五、怎么实现进程有自己独立的Rootfs?

5.1 容器这个进程为什么有自己的Rootfs?

  1.         因为容器镜像文件系统驱动overlay2以及Mount namespace结合使用来实现进程拥有自己namespace级别的的Rootfs.

        首先容器镜像是基于层级,一层一层只读层叠加最后合并组合为一个镜像。技术叫做联合文件系统,驱动名称为overlay2,在主机上的呈现为一个一个独立的目录。(每一层镜像可读层就是一个目录),编写dockerfile中每一个命令。RUN、COPY、CMD都是一层。

六、容器是怎么挂载和使用主机的Dir的?

        当容器启动时,会使用overlay2存储驱动将容器的LowerDir、UpperDir、WorkDir、MergedDir镜像合并,组成容器的Rootfs(进入容器后看到的就是Rootfs文件系统加上隔离的namespace容器的用户空间),并且在 Docker 中,当你使用 -v 或 --volume 参数将主机上的目录挂载到容器中时,Docker 实际上会在容器的 mount namespace 中创建一个新的挂载点,将主机上的目录挂载到这个新的挂载点上。这样,容器内的进程就能够通过这个挂载点来访问主机上的目录,实现文件共享和交互。

七、Docker 容器运行时组件的简要描述:

1. **Docker Client(Docker 客户端):**
   - Docker 客户端是用户与 Docker 交互的主要方式之一,它可以通过命令行或者 API 来与 Docker 交互。
   - Docker 客户端可以连接到 Docker 守护进程(Dockerd)并发送命令来管理容器、镜像、网络等资源。
   - Docker 客户端可以与 Docker 守护进程通过 REST API 或者 Unix 套接字进行通信。

2. **Dockerd(Docker 守护进程):**
   - Dockerd 是 Docker 的守护进程,负责管理容器的生命周期、镜像的存储和分发、网络的管理等。
   - Dockerd 通过监听 REST API 的请求来接收来自 Docker 客户端的命令,并执行相应的操作。
   - Dockerd 也可以与其他容器运行时组件,如 Containerd,进行交互。

3. **Containerd:**
   - Containerd 是一个用于管理容器生命周期的守护进程,是 Docker 中的一个核心组件。
   - Containerd 负责管理容器的创建、运行、暂停、停止、删除等操作,以及容器与宿主机资源的隔离。
   - Containerd 提供了 REST API,允许其他组件(如 Dockerd)通过 REST API 调用它的功能。

4. **Runc:**
   - Runc 是一个用于运行容器的工具,是 Open Container Initiative(OCI)规范的实现之一。
   - Runc 负责根据 OCI 规范创建和运行容器,它使用 Linux 的原生功能,如 cgroups、namespace 等来实现容器的隔离和运行。
   - Dockerd 使用 Runc 来启动容器,它通过调用 Runc 的接口来创建和管理容器的生命周期。

       下面是创建容器各个组件的交付:

  • Docker Client 使用 Unix 套接字或者 REST API 与 Dockerd 守护进程通信,向 Dockerd 发送命令以管理容器和其他 Docker 资源。

  • Dockerd 守护进程接收来自 Docker Client 的命令,并根据这些命令调用 Containerd 来管理容器的生命周期。

  • Containerd 负责实际的容器管理,包括创建、运行、停止、删除等操作。它通过 REST API 与 Dockerd 通信,接收来自 Dockerd 的命令。

  • 重点:

  • 当 Containerd 需要创建容器时,它会调用 Runc 来实际运行容器。Runc 使用系统调用(如 clone())来创建新的容器进程,并根据需要分配新的命名空间等资源。

八、虽然容器和主机使用不同的namespace,但他们使用相同的主机上的内核空间,就以为着他们共享主机上的物理计算资源。怎么做限制的?

        8.1 Cgroup简单介绍   

Cgroup(Control Groups)是 Linux 内核提供的一个功能,用于管理和限制进程对系统资源的使用。它主要用于计费、监视和限制进程对 CPU、内存、存储 IO、网络 IO 等资源的使用。

Cgroup 使用树状结构来组织资源的管理,每个分支代表一个子系统,而子系统则代表一个特定的资源,如 CPU、内存、存储 IO、网络 IO 等。进程被分配到适当的子系统中,以便对其资源使用进行管理。

在 Cgroup 中,每个进程都被保存在一个名为 tasks 的文件中,该文件位于相应子系统的目录下。通过编辑和配置 /sys/fs/cgroup 目录下的文件,可以对子系统和进程的资源使用进行调整和控制。

        8.2 那么默认情况下新创建的容器是不受资源限制的吗?

默认情况下,新创建的容器是不受资源限制的。当你使用 Docker 创建一个容器时,Docker 默认不会为该容器设置任何资源限制,容器可以使用主机上的所有资源,包括 CPU、内存、磁盘 IO、网络带宽等。

要为容器设置资源限制,你可以通过 Docker 的参数来指定容器的资源限制,如 --cpu--memory--cpus 等。你还可以使用 Docker Compose 文件或者 Kubernetes 的资源配置来设置容器的资源限制。通过设置这些参数,你可以限制容器使用的 CPU 资源数量、内存使用量等,以防止容器占用过多的系统资源。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/25658.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

刷代码随想录有感(99):动态规划——使用最小花费爬楼梯

题干&#xff1a; 代码&#xff1a; class Solution { public:int minCostClimbingStairs(vector<int>& cost) {vector<int>dp(cost.size() 1);dp[0] 0;dp[1] 0;for(int i 2; i < cost.size(); i){dp[i] min(dp[i - 1] cost[i - 1], dp[i - 2] cost…

Leetcode 力扣114. 二叉树展开为链表 (抖音号:708231408)

给你二叉树的根结点 root &#xff0c;请你将它展开为一个单链表&#xff1a; 展开后的单链表应该同样使用 TreeNode &#xff0c;其中 right 子指针指向链表中下一个结点&#xff0c;而左子指针始终为 null 。展开后的单链表应该与二叉树 先序遍历 顺序相同。 示例 1&#xf…

KUKA机器人中断编程详细教程1—了解中断

在公众号查看更多内容。 在KUKA机器人编程与调试中&#xff0c;经常会用到中断编程。通过中断实现机器人暂停&#xff0c;或者停止当前的动作进入中断后的程序中接着运行&#xff0c;以此来满足实际的调试要求。 1、中断的概念 ①当出现诸如输入等定义的事件时&#xff0c;…

【算法篇】求最长公共前缀JavaScript版本

题目描述 给你一个大小为 n 的字符串数组 strs &#xff0c;其中包含n个字符串 , 编写一个函数来查找字符串数组中的最长公共前缀&#xff0c;返回这个公共前缀。 数据范围&#xff1a; 数据范围:0<n<5000&#xff0c;0<len(strsi)< 5000 进阶:空间复杂度 O(1)&a…

Typora Markdown编辑器 for Mac v1.8.10 安装

Mac分享吧 文章目录 效果一、准备工作二、开始安装1、双击运行软件&#xff0c;将其从左侧拖入右侧文件夹中&#xff0c;等待安装完毕2. 应用程序显示软件图标&#xff0c;表示安装成功 三、运行调试1、修改主题2、显示文档列表&#xff0c;如下图3、查看版本信息 **安装完成&…

【PR2019】怎样批量添加转场效果及修改默认持续时间

一&#xff0c;设置“交叉溶解”效果到所有素材 选择效果&#xff0c;右击“将所选过渡设置为默认过渡”&#xff1a; 框选所有素材&#xff0c;“Ctrl D”&#xff1a; 每个素材中间有有了交叉溶解的效果&#xff1a; 二&#xff0c;修改效果属性 2.1&#xff0c;单个修…

北航第五次数据结构与程序设计编程题复习

北航第五次数据结构与程序设计编程题复习 树叶节点遍历&#xff08;树-基础题&#xff09;计算器&#xff08;表达式计算-表达式树实现&#xff09;服务优化词频统计&#xff08;树实现&#xff09; 树叶节点遍历&#xff08;树-基础题&#xff09; 【问题描述】 从标准输入中…

CTFHUB-SQL注入-报错注入

目录 报错注入概述 报错注入的原理 报错注入的步骤 报错注入的常用函数 实战案例 结论 方法1&#xff1a;updatexml函数 查看数据库名 查看表名 查看表中数据 方法2&#xff1a;extractvalue函数 查看数据库名 查看数据库中的表名 查看表中字段名 查看表中数据 报…

第2回 从0x7c00到0x90000

将数据段寄存器ds的值变成了0x07c0,方便了之后访问内存时利用这个段基址进行寻址,接下来,我们带着这两行代码继续往下看6行: 此时ds寄存器的值已经是0x07c0了,然后用同样的方式将es寄存器的值变成0x9000,接着又把cs寄存器的值变成256。 好的,此时ds,es,cx寄存器的值,都…

CleanMyMac2024最新免费电脑Mac系统优化工具

大家好&#xff0c;我是你们的好朋友——软件评测专家&#xff0c;同时也是一名技术博主。今天我要给大家种草一个超级实用的Mac优化工具——CleanMyMac&#xff01; 作为一个长期使用macOS的用户&#xff0c;我深知系统运行时间长了&#xff0c;缓存文件、日志、临时文件等都会…

LLVM Cpu0 新后端5 静态重定位 动态重定位

想好好熟悉一下llvm开发一个新后端都要干什么&#xff0c;于是参考了老师的系列文章&#xff1a; LLVM 后端实践笔记 代码在这里&#xff08;还没来得及准备&#xff0c;先用网盘暂存一下&#xff09;&#xff1a; 链接: https://pan.baidu.com/s/1yLAtXs9XwtyEzYSlDCSlqw?…

每日一题——Python实现PAT乙级1019 数字黑洞(举一反三+思想解读+逐步优化)

一个认为一切根源都是“自己不够强”的INTJ 个人主页&#xff1a;用哲学编程-CSDN博客专栏&#xff1a;每日一题——举一反三Python编程学习Python内置函数 Python-3.12.0文档解读 目录 我的写法 点评代码的优缺点&#xff1a; 时间复杂度&#xff1a; 空间复杂度&#…

【类型商店】字符字符串(下)

啊&#xff0c;哈喽&#xff0c;小伙伴们大家好。我是#Y清墨&#xff0c;今天呐&#xff0c;我要介绍的是字符与字符串。 导语 前两期&#xff0c;我们已经懂得了概念&#xff0c;今天来看些函数。 正题 一.增加或连接 &#xff08;1) 后面增加() string s1,s2; //定义 s…

Meta首席AI科学家Yann LeCun指出生成式AI的不足

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

express入门01服务器搭建以及get和post请求的监听

微搭提供了后端API的能力&#xff0c;但是不同的版本收费差别巨大&#xff0c;因为使用的门槛限制了中小企业使用低代码平台。那可不可以既要又要呢&#xff1f;答案是肯定的&#xff0c;那其实掌握一定的后端框架&#xff0c;借助我们在低代码中已经熟练掌握的技能其实是比较容…

OpenCV学习(4.9) OpenCV中的轮廓

1.目标 了解轮廓是什么。学习寻找轮廓&#xff0c;绘制轮廓等您将看到以下功能&#xff1a;**cv.findContours()** &#xff0c;**cv.drawContours()* 2.什么是轮廓 轮廓可以简单地解释为连接具有相同颜色或强度的所有连续点(沿边界)的曲线。轮廓是用于形状分析以及对象检测…

友情提示:lazarus的tsortgrid.autofillcolumns存在BUG

直接在tsortgrid的属性中设置autofillcolumns为true&#xff0c;会提示&#xff1a;123个错误。即使修改为false&#xff0c;编译运行照样会出现上述错误。唯一解决的办法就是删除sortgrid重新添加一个。 代码设置SortGrid1.AutoFillColumns : TRUE不受影响。

【Windows】UWP - Application Frame 窗口句柄溯源

目录 一、问题描述 二、解决方案 三、测试代码 参考文献 本文出处链接&#xff1a;[https://blog.csdn.net/qq_59075481/article/details/139574981]。 一、问题描述 当 GUI 线程的窗口属于 Windows/UWP 应用程序时&#xff0c;它们始终由进程 ApplicationFrameHost 托管…

量化投资分析平台 迅投 QMT(六)资产定价绕不过去的BSM模型

量化投资分析平台 迅投 QMT [迅投 QMT](https://www.xuntou.net/?user_code7NYs7O)我目前在使用什么是BSM模型CQF课程介绍模型的五个重要的假设模型公式 我们为啥要学&#xff08;知道&#xff09;这玩意儿呢&#xff1f;隐含波动率&#xff08;Implied Volatility&#xff09…

初阶 《函数》 4. 函数的调用

4. 函数的调用 4.1 传值调用 函数的形参和实参分别占有不同内存块&#xff0c;对形参的修改不会影响实参 4.2 传址调用 传址调用是把函数外部创建变量的内存地址传递给函数参数的一种调用函数的方式 这种传参方式可以让函数和函数外边的变量建立起真正的联系&#xff0c;也就是…