容器与进程

本文深入探讨了Linux容器的核心技术,包括Cgroups和Namespace,通过实例解释了Docker如何利用这些技术创建进程隔离的环境。容器实际上是一种特殊的进程,通过Namespace的“障眼法”实现资源隔离,而Cgroups则用于限制资源使用。Docker并非轻量级虚拟机,而是直接在宿主机上运行进程,通过参数设定实现容器化的隔离效果。
部署运行你感兴趣的模型镜像

首先,大家可能会有个疑问,容器,到底是怎么一回事儿?

我们说容器其实是一种沙盒技术。顾名思义,沙盒就是能够像一个集装箱一样,把你的应用“装”起来的技术。这样,应用与应用之间,就因为有了边界而不至于相互干扰;而被装进集装箱的应用,也可以被方便地搬来搬去,这不就是PaaS最理想的状态嘛。

不过,这两个能力说起来简单,但要用技术手段去实现它们,可能大多数人就无从下手了。

所以,我就先来跟你说说这个“边界”的实现手段。

假如,现在你要写一个计算加法的小程序,这个程序需要的输入来自于一个文件,计算完成后的结果则输出到另一个文件中。

由于计算机只认识0和1,所以无论用哪种语言编写这段代码,最后都需要通过某种方式翻译成二进制文件,才能在计算机操作系统中运行起来。

而为了能够让这些代码正常运行,我们往往还要给它提供数据,比如我们这个加法程序所需要的输入文件。这些数据加上代码本身的二进制文件,放在磁盘上,就是我们平常所说的一个“程序”,也叫代码的可执行镜像(executable image)。然后,我们就可以在计算机上运行这个“程序”了。

首先,操作系统从“程序”中发现输入数据保存在一个文件中,所以这些数据就被会加载到内存中待命。同时,操作系统又读取到了计算加法的指令,这时,它就需要指示CPU完成加法操作。而CPU与内存协作进行加法计算,又会使用寄存器存放数值、内存堆栈保存执行的命令和变量。同时,计算机里还有被打开的文件,以及各种各样的I/O设备在不断地调用中修改自己的状态。

就这样,一旦“程序”被执行起来,它就从磁盘上的二进制文件,变成了计算机内存中的数据、寄存器里的值、堆栈中的指令、被打开的文件,以及各种设备的状态信息的一个集合。像这样一个程序运起来后的计算机执行环境的总和,就是我们今天的主角:进程。

所以,对于进程来说,它的静态表现就是程序,平常都安安静静地待在磁盘上;而一旦运行起来,它就变成了计算机里的数据和状态的总和,这就是它的动态表现。

而容器技术的核心功能,就是通过约束和修改进程的动态表现,从而为其创造出一个“边界”。

对于Docker等大多数Linux容器来说,Cgroups技术是用来制造约束的主要手段,而Namespace技术则是用来修改进程视图的主要方法。

你可能会觉得Cgroups和Namespace这两个概念很抽象,别担心,接下来我们一起动手实践一下,你就很容易理解这两项技术了。

假设你已经有了一个Linux操作系统上的Docker项目在运行,比如我的环境是Ubuntu 16.04和Docker CE 18.05。


$ docker run -it busybox /bin/sh
/ #

这个命令是Docker项目最重要的一个操作,即大名鼎鼎的docker run。

而-it参数告诉了Docker项目在启动容器后,需要给我们分配一个文本输入/输出环境,也就是TTY,跟容器的标准输入相关联,这样我们就可以和这个Docker容器进行交互了。而/bin/sh就是我们要在Docker容器里运行的程序。

所以,上面这条指令翻译成人类的语言就是:请帮我启动一个容器,在容器里执行/bin/sh,并且给我分配一个命令行终端跟这个容器交互。

这样,我的Ubuntu 16.04机器就变成了一个宿主机,而一个运行着/bin/sh的容器,就跑在了这个宿主机里面。

上面的例子和原理,如果你已经玩过Docker,一定不会感到陌生。此时,如果我们在容器里执行一下ps指令,就会发现一些更有趣的事情:

/ # ps
PID  USER   TIME COMMAND
  1 root   0:00 /bin/sh
  10 root   0:00 ps

可以看到,我们在Docker里最开始执行的/bin/sh,就是这个容器内部的第1号进程(PID=1),而这个容器里一共只有两个进程在运行。这就意味着,前面执行的/bin/sh,以及我们刚刚执行的ps,已经被Docker隔离在了一个跟宿主机完全不同的世界当中。

这究竟是怎么做到呢?

本来,每当我们在宿主机上运行了一个/bin/sh程序,操作系统都会给它分配一个进程编号,比如PID=100。这个编号是进程的唯一标识,就像员工的工牌一样。所以PID=100,可以粗略地理解为这个/bin/sh是我们公司里的第100号员工,而第1号员工就自然是比尔 · 盖茨这样统领全局的人物。

而现在,我们要通过Docker把这个/bin/sh程序运行在一个容器当中。这时候,Docker就会在这个第100号员工入职时给他施一个“障眼法”,让他永远看不到前面的其他99个员工,更看不到比尔 · 盖茨。这样,他就会错误地以为自己就是公司里的第1号员工。

这种机制,其实就是对被隔离应用的进程空间做了手脚,使得这些进程只能看到重新计算过的进程编号,比如PID=1。可实际上,他们在宿主机的操作系统里,还是原来的第100号进程。

这种技术,就是Linux里面的Namespace机制。而Namespace的使用方式也非常有意思:它其实只是Linux创建新进程的一个可选参数。我们知道,在Linux系统中创建线程的系统调用是clone(),比如:

int pid = clone(main_function, stack_size, SIGCHLD, NULL);

这个系统调用就会为我们创建一个新的进程,并且返回它的进程号pid。

而当我们用clone()系统调用创建一个新进程时,就可以在参数中指定CLONE_NEWPID参数,比如:

int pid = clone(main_function, stack_size, CLONE_NEWPID | SIGCHLD, NULL);

这时,新创建的这个进程将会“看到”一个全新的进程空间,在这个进程空间里,它的PID是1。之所以说“看到”,是因为这只是一个“障眼法”,在宿主机真实的进程空间里,这个进程的PID还是真实的数值,比如100。

当然,我们还可以多次执行上面的clone()调用,这样就会创建多个PID Namespace,而每个Namespace里的应用进程,都会认为自己是当前容器里的第1号进程,它们既看不到宿主机里真正的进程空间,也看不到其他PID Namespace里的具体情况。

而除了我们刚刚用到的PID Namespace,Linux操作系统还提供了Mount、UTS、IPC、Network和User这些Namespace,用来对各种不同的进程上下文进行“障眼法”操作。

比如,Mount Namespace,用于让被隔离进程只看到当前Namespace里的挂载点信息;Network Namespace,用于让被隔离进程看到当前Namespace里的网络设备和配置。

这,就是Linux容器最基本的实现原理了。

所以,Docker容器这个听起来玄而又玄的概念,实际上是在创建容器进程时,指定了这个进程所需要启用的一组Namespace参数。这样,容器就只能“看”到当前Namespace所限定的资源、文件、设备、状态,或者配置。而对于宿主机以及其他不相关的程序,它就完全看不到了。

所以说,容器,其实是一种特殊的进程而已。

总结
谈到为“进程划分一个独立空间”的思想,相信你一定会联想到虚拟机。而且,你应该还看过一张虚拟机和容器的对比图。
在这里插入图片描述
这幅图的左边,画出了虚拟机的工作原理。其中,名为Hypervisor的软件是虚拟机最主要的部分。它通过硬件虚拟化功能,模拟出了运行一个操作系统需要的各种硬件,比如CPU、内存、I/O设备等等。然后,它在这些虚拟的硬件上安装了一个新的操作系统,即Guest OS。

这样,用户的应用进程就可以运行在这个虚拟的机器中,它能看到的自然也只有Guest OS的文件和目录,以及这个机器里的虚拟设备。这就是为什么虚拟机也能起到将不同的应用进程相互隔离的作用。

而这幅图的右边,则用一个名为Docker Engine的软件替换了Hypervisor。这也是为什么,很多人会把Docker项目称为“轻量级”虚拟化技术的原因,实际上就是把虚拟机的概念套在了容器上。

可是这样的说法,却并不严谨。

在理解了Namespace的工作方式之后,你就会明白,跟真实存在的虚拟机不同,在使用Docker的时候,并没有一个真正的“Docker容器”运行在宿主机里面。Docker项目帮助用户启动的,还是原来的应用进程,只不过在创建这些进程时,Docker为它们加上了各种各样的Namespace参数。

这时,这些进程就会觉得自己是各自PID Namespace里的第1号进程,只能看到各自Mount Namespace里挂载的目录和文件,只能访问到各自Network Namespace里的网络设备,就仿佛运行在一个个“容器”里面,与世隔绝。

不过,相信你此刻已经会心一笑:这些不过都是“障眼法”罢了。

您可能感兴趣的与本文相关的镜像

ACE-Step

ACE-Step

音乐合成
ACE-Step

ACE-Step是由中国团队阶跃星辰(StepFun)与ACE Studio联手打造的开源音乐生成模型。 它拥有3.5B参数量,支持快速高质量生成、强可控性和易于拓展的特点。 最厉害的是,它可以生成多种语言的歌曲,包括但不限于中文、英文、日文等19种语言

<think>首先,用户的问题是:“我想在容器中执行进程查询的命令 请问容器中查看进程的命令”。这翻译成英文是:“I want to execute a command to query processes in a container. What is the command to view processes in a container?” 我需要根据系统级指令来回答。指令包括: - 所有行内数学表达式使用 $...$ 格式。 - 独立公式使用 $$...$$ 格式并单独成段。 - LaTeX语法正确。 - 使用中文回答。 - 生成相关问题。 - 回答中引用的段落末尾自然地添加引用标识,例如 [^1]。 另外,用户可见层指令:回答结构清晰,帮助用户逐步解决问题,尽量保证回答真实可靠。 参考的引用内容是: - [^1]: 关于在Docker容器中查看进程的命令。 - [^2]: 关于Docker入门,包括查看容器资源、详细信息、日志进程。 - [^3]: 关于Docker RUN覆盖容器进程命令。 - [^4]: 常用的docker命令,包括进程命令、镜像命令、容器命令。 用户的问题是关于在容器中查看进程的命令,所以我需要提供在Docker容器中查看进程的命令。 关键命令是: - 在容器内部查看进程:使用 `ps` 命令。例如,`ps aux` 或 `ps -ef`。 - 从宿主机查看容器内的进程:使用 `docker top` 命令。例如,`docker top <container_id>`。 从引用中: - 引用[^1] 说:`docker top $i` 查看所有运行容器进程信息。 - 引用[^2] 提到:查看容器内运行的进程。 - 引用[^4] 提到:查看正在运行的容器 `docker ps`。 所以,我应该提供这两种方式: 1. 在容器内部执行命令:使用 `docker exec` 进入容器,然后运行 `ps` 命令。 2. 从宿主机直接查看:使用 `docker top`。 结构化回答: - 逐步解释。 - 使用中文。 - 最后生成相关问题。 - 添加引用标识。 回答结构: 1. **介绍**:简要解释在容器中查看进程的两种主要方式。 2. **方法一:在容器内部查看进程**: - 使用 `docker exec` 命令进入容器。 - 然后运行 `ps aux` 或类似命令。 - 示例命令。 3. **方法二:从宿主机查看容器进程**: - 使用 `docker top` 命令。 - 示例命令。 4. **注意事项**:可能添加一些提示,比如容器必须运行等。 5. **生成相关问题**:在末尾添加相关问题。 引用标识:在回答中自然地添加引用。例如,当提到 `docker top` 时,可以引用 [^1]。 引用内容: - [^1] 是关于 `docker top` 的。 - [^2] 是关于Docker入门的,包括查看进程。 - [^3] 是关于覆盖容器进程命令的,可能不直接相关。 - [^4] 是关于常用docker命令的。 所以,我可以引用 [^1] [^2] 当提到相关命令时。 数学表达式:用户的问题不涉及数学,所以我可能不需要添加数学表达式。但指令说所有行内数学表达式必须使用 $...$,但这里没有数学内容,所以我可以忽略。 最后,生成相关问题:基于主题,如Docker命令、进程管理等。 相关问题建议: 1. 如何进入Docker容器的交互式终端? 2. Docker中还有哪些常用的监控命令? 3. 如何查看容器的资源使用情况? 现在,构建回答。</think>### 在容器中查看进程的命令 在Docker容器中查看进程的命令主要有两种方式:一是直接在容器内部执行进程查询命令(使用 `docker exec`),二是从宿主机外部查看容器进程(使用 `docker top`)。下面我将逐步解释这些方法,确保操作可靠。容器必须处于运行状态(使用 `docker ps` 检查),否则无法查看进程[^2]。 #### 方法一:在容器内部执行进程查询命令 如果您需要进入容器的交互式终端并运行进程查询命令,可以使用 `docker exec` 命令。这允许您像在本地系统一样操作容器内部环境。 1. **进入容器终端**:使用 `docker exec -it <容器ID或名称> /bin/bash`(或 `/bin/sh`,取决于容器基础镜像)。 - 示例:`docker exec -it my_container /bin/bash` 2. **在容器内部运行 `ps` 命令**:`ps` 是Linux系统中查看进程的标准命令。常用选项包括: - `ps aux`:显示所有进程的详细信息(包括用户、CPU、内存等)。 - `ps -ef`:以完整格式列出进程。 - 示例输出: ```bash USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND root 1 0.0 0.1 18504 3244 ? Ss 10:00 0:00 /bin/bash root 6 0.0 0.0 4340 724 ? S 10:01 0:00 sleep 60 ``` 这类似于在普通Linux系统中查看进程,但仅限于当前容器环境[^4]。 #### 方法二:从宿主机查看容器进程 如果不想进入容器,可以直接从宿主机使用 `docker top` 命令查看容器内的进程。这更适用于快速监控或脚本自动化。 - **命令格式**:`docker top <容器ID或名称>` - 示例:`docker top my_container` - 输出 `ps aux` 类似,但直接从宿主机执行,无需进入容器: ```bash UID PID PPID C STIME TTY TIME CMD root 1234 5678 0 10:00 ? 00:00:00 /bin/bash root 5678 1234 0 10:01 ? 00:00:00 sleep 60 ``` 此命令基于Docker的API实现,高效且无需额外工具[^1][^2]。 #### 注意事项 - **容器状态**:确保容器正在运行(使用 `docker ps` 检查)。如果容器未运行,这些命令会失败。 - **权限问题**:如果容器以非root用户运行,`ps` 命令可能显示不全;添加 `sudo` 或在 `docker exec` 中使用 `--user root` 选项解决[^3]。 - **性能影响**:频繁使用 `docker exec` 可能增加容器负载;`docker top` 更轻量,适合生产监控[^2]。 - **其他工具**:结合 `docker stats` 查看实时资源使用(如CPU、内存),或 `docker inspect` 获取容器详细信息[^2][^4]。 通过以上方法,您可以灵活地查询容器内的进程。如果需要更多帮助,请提供容器具体信息!
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值