一探Linux下的七大进程状态

news2025/1/4 17:13:16

在这里插入图片描述

文章目录

  • 一、前言
  • 二、操作系统学科下的进程状态
    • 1、运行状态
    • 2、阻塞状态
    • 3、挂起状态
  • 三、Linux下的7种进程状态
    • 1、运行状态R
    • 2、浅度睡眠状态S
    • 3、深度睡眠状态D
        • 一场有趣的官司
    • 4、停止状态T
    • 5、进程跟踪状态t
    • 6、死亡状态X
    • 7、僵死状态Z —— 两个特殊进程
      • ① 僵尸进程
      • ② 孤儿进程
  • 四、总结与提炼

一、前言

Hello,大家好,本文我们所要介绍的是有关Linux下的进程状态

  • 在上一文中,我们重点介绍了有关 Linux下进程的基本概念,了解了什么是进程、怎么去描述并组织进程、创建一个进程。
  • 在本文中,我们将先通过了解操作系统学科下的进程状态,对进程的状态有一个基本的概念,然后呢再去学习Linux下的7种进程状态,学习这PCB结构体中的第二个成员变量

task_ struct内容分类

  • 标示符: 描述本进程的唯一标示符,用来区别其他进程。
  • 状态: 任务状态,退出代码,退出信号等。
  • 优先级: 相对于其他进程的优先级。
  • 程序计数器: 程序中即将被执行的下一条指令的地址。
  • 内存指针: 包括程序代码和进程相关数据的指针,还有和其他进程共享的内存块的指针
  • 上下文数据: 进程执行时处理器的寄存器中的数据[休学例子,要加图CPU,寄存器]。
  • I/O状态信息: 包括显示的I/O请求,分配给进程的I/O设备和被进程使用的文件列表。
  • 记账信息: 可能包括处理器时间总和,使用的时钟数总和,时间限制,记账号等。
  • 其他信息

二、操作系统学科下的进程状态

  • 对于进程而言呢,它是操作系统中的概念,如果有学习过《操作系统》这门学科的话应该可以很清楚对于进程而言的话是存在着许许多多的状态,如果一开始刚刚接触的小伙伴一定会感觉这么多状态要怎么区分呀😵

在这里插入图片描述
其实那么多的状态,真正主要的也就那么几个,所以接下去我会中重点讲解以下几种进程的状态

1、运行状态

首先我们要谈到的是【运行状态】,这个状态是最普遍的

  • 首先对于一个进程而言,我们知道它是由 内核数据结构 + 所对应的代码和数据 所组成的,所以当系统中存在多个进程的时候就势必会存在多个结构体;当然我们需要将这些进程给链接组织起来
  • 那么这些进程就相当于是在处在一个运行队列中,我们如果要找到这个队列中的某个进程的话,只需要找到这个进程的头部即可,那我们就可以对应地去调度某个进程,把这个进程所对应的代码和数据放到CPU上去执行

💬 因为每个进程是需要去竞争CPU资源的,但是呢CPU不可能同时给这么多进程分配资源

  • 所以每一个CPU都会去维护一个运行队列,里面的队头指针head所指向就是第一个进程所对应的【task_struct】,队尾指针tail所指向就是最后一个所对应的【task_struct】。所以我们要运行某一个进程只需要将 head 所指向的那个进程放到CPU上去运行即可

在这里插入图片描述

  • 但是CPU无法直接去找到某个需要调度的进程来运行,此时呢就需要一种东西叫做【调度器】,CPU通过找到运行队列,运行队列找到调度器,然后调度器再去一一调度所需要运行的进程,那么此时所被调度的处于运行队列里的这些进程所处的状态我们称之为 运行状态R

在这里插入图片描述

提问:一个进程只要把自己放到CPU上开始运行了,是不是一直要到执行完毕,才把自己放下来?

  • 不是,每一个进程都有一个叫做:时间片的概念! 其时间大概是在10 ms左右。所以并不是一个进程一直在执行,而是这多个进程在一个时间段内所有代码都会被执行 —— 这就叫做【并发执行】
  • 所以呢这就一定会存在大量的进程被CPU放上去、拿下来的动作 —— 这就叫做【进程切换】

💬 所以呢我们不要拿自己的时间感受去衡量CPU,其运行一遍速度是非常快的,你根本感受不到这种进程切换的效果

2、阻塞状态

在介绍完【运行状态】后,我们再来讲讲【阻塞状态】

  • 之前我们在讲 操作系统基本概念 的时候,有说到过操作系统要去管理底层的硬件的话需要进行【先描述,再组织】的动作,那就是将这一个个生冷的硬件抽象成为结构体类型,内部的type代表这是一个什么类型的硬件、status代表这个硬件当前在操作系统中所处的状态是什么样的,wait_queue代表的则是等待队列的指针

在这里插入图片描述

  • 那我们继续去谈起操作系统中的【进程】,现在呢有一个进程想要去读取键盘中的数据,那么它就需要与我们刚才所描述的结构体相互勾连起来,因为操作系统根据这个结构体顺着体系结构去访问到最底层的硬件需要实现,所以这个进程便需要一直处于等待状态,我们可以将其链入【等待队列】中,即我们刚才所讲的wait_queue

💬 那有同学可可能会疑惑这为什么叫做【等待队列】呢?明显只有一个进程鸭🦆

  • 一个进程当然不能算是队列,若此时又有一个进程也要来读取键盘中数据的话,我们就需要将其链入到这个等待队列中。此时这两个进程所处的状态即为 阻塞状态

在这里插入图片描述

  • 那么当键盘里一旦有数据了,我们只需要把这个进程放到我们在上面所讲的【运行队列】中即可,那么CPU在调度的时候就可以自动地到底层设备里去读取了;如果当键盘里没有输入的时候,最终我们的进程就只能在每一个各自的等待队列里去等待,把这种处于等待队列的进程称之为 阻塞状态

在这里插入图片描述

3、挂起状态

最后我们再来讲讲一种状态叫做【挂起状态】

  • 对于操作系统而言,我们知道它要为当前的所有进程分配内存资源,在上面我只画了一个等待队列,而且只有两个进程,若是一个等待队列中存在多个进程,并且通过有多个等待队列在排队的话,操作系统中的内存资源就会出现【消耗殆尽】的问题
  • 那么操作系统就得保证在正常情况下省出来内存资源

在这里插入图片描述

要怎么去省呢?

  • 只要一个进程没有被调度(运行),那么当前这个进程的 代码和数据 就是空闲的、没有被使用的。于是操作系统就想办法把这个进程内核的PCB给保留,然后将其 代码和数据 重新交换到外设当中【换出】,那么此时只有一个PCB在这里排队,我们把上面这种状态就称之为 挂起状态

在这里插入图片描述

  • 当这个进程就绪了,把这个进程放到运行队列时,此时再考虑把 代码和数据 重新放回进程中【换入】

💬 那有的同学说:为什么要这样去做呢?这样做有什么意义?

  • 答:当一个进程所对应的代码和数据交换到外设上时,操作系统内部就腾出了相应的空间,此时OS就可以将这块多出来的空间重新给到其他需要的进程使用,从而达到了循环利用系统资源的目的,节省了系统开销。

三、Linux下的7种进程状态

在介绍完操作系统学科下的三种最主要进程状态后,我们对进程的状态有了基本的概念,接下去就让我们正式地来学习一下Linux系统下7种进程状态

先来小结并回顾一下上面所学:

  1. 如果当前是【运行状态】,那么接下来就需要被调度运行
  2. 如果当前是【阻塞状态】,那就等条件就绪,等设备准备好就把当前进程投递到运行队列里,然后再被CPU调度运行
  3. 如果当前是【挂起状态】,要做的就是把当时换出的代码和数据重新换入,然后再把所对应的进程列入到运行队列中

以下就是关于进程的所有状态

static const char * const task_state_array[] = {
"R (running)", /* 0 */
"S (sleeping)", /* 1 */
"D (disk sleep)", /* 2 */
"T (stopped)", /* 4 */
"t (tracing stop)", /* 8 */
"X (dead)", /* 16 */
"Z (zombie)", /* 32 */
};

1、运行状态R

首先我们要来聊的是【运行状态R】

  • 来看下下面的这段代码,是一个死循环去printf打印循环语句
  1 #include <stdio.h>
  2 #include <unistd.h>
  3 
  4 int main(void)
  5 {
  6     while(1);                                                                
  7     {
  8         printf("hello bit\n");
  9     }
 10 
 11     return 0;
 12 }
  • 然后我们将下面的代码给运行起来,观察这个进程的状态时,看到其显示为S+,不过呢读者想看到的应该是R才对

在这里插入图片描述

  • 接下去呢,我们把代码修改一下再来看看,不使用printf打印语句了,而且直接使用一个while(1)去做循环
  1 #include <stdio.h>
  2 #include <unistd.h>
  3 
  4 int main(void)
  5 {
  6     while(1);                                                                   
  7     //{
  8     //    printf("hello bit\n");
  9     //}
 10 
 11     return 0;
 12 }
  • 然后我们看到,此时再运行起来时这个进程的状态就改变了,变成了R+,这才是我们想要的【进程状态】

在这里插入图片描述
💬 那有读者就要问了:为什么把printf打印语句给去掉之后就变成这样了呢?

  • 原因就在于printf打印语句它是属于IO流的一种,第一次因为是循环的缘故,它一直在等IO设备就绪,所以其进程状态就一直为S+,对应的即是在操作系统中的【阻塞状态】;但是当我们去掉printf这种IO流之后呢,它就是在纯纯运行,没有IO,那也就变成了R状态

这里再补充说明一下这个SR后面的+

  • 这里的R+代表的就是这个进程是在前台运行的,所以我们在输入任何指令后不会对其造成 任何的影响

在这里插入图片描述

  • 那若是我们不以正常的方式去启动这个进程的话,其进程的状态就会不一样了,可以看到我在./myproc的后面加上了一个&;那么其状态变成了R,此代表的意思就是这个进程它是运行在了【后台】的

在这里插入图片描述

2、浅度睡眠状态S

接下去我们再来介绍一下Linux下的睡眠状态S

  • 下面我们要运行的代码是每隔一秒打印一句内容
  1 #include <stdio.h>
  2 #include <unistd.h>
  3 
  4 int main(void)
  5 {
  6     while(1)
  7     {
  8         sleep(1);                                                                   
  9         printf("hello bit\n");
 10     }
 11 
 12     return 0;
 13 }
  • 运行起来后可以看到这个进程所处的状态即为S+睡眠状态

在这里插入图片描述
💬 那有同学就很疑惑,这个进程不是在运行吗?为什么不是R状态呢?

  • 我们可以通过动图来观察看看,每一次去查看这个进程的状态时,都是处于S+这个睡眠状态。其实呢,这和外设与CPU的速度之差是有关系的,CPU那个运行速度是极其快速的
  • 虽然我们肉眼可见右侧的进程是一直在每隔一秒运行,但是呢在我们查进程状态的时候看到的却只能是S+;因为有99.99%的时间都在进行等待,而只有0.01%的时间在运行,它们之间简直可以说是数量级别的差别

在这里插入图片描述

  • 不过呢,上面这样还无法看出一个进程是否真正地处于睡眠状态,我们再通过一段代码来看看
  1 #include <stdio.h>
  2 #include <unistd.h>
  3 
  4 int main(void)
  5 {
  6     int a = 0;
  7     printf("Enter# ");
  8     scanf("%d", &a);
  9 
 10     printf("echo : %d\n", a);
 11     return 0;                                                                         
 12 } 
  • 将该进程运行起来我们可以看到其是出于S+的状态,因为【shell】此时正在等待用户的输入,这个就对应到了我们上面所讲到的 阻塞状态

在这里插入图片描述

在这里插入图片描述

  • 不仅如此,像我们一直在使用的bash,也可以算是一种【阻塞状态】,一直等待着我们去输入命令行,一旦有了的话它就进行读取

在这里插入图片描述

在这里插入图片描述

3、深度睡眠状态D

除了【浅度睡眠】之外呢,还有一种叫做【深度睡眠】,它们俩呢,都是 阻塞状态

  • 对于浅度睡眠来说,之所以称为 “浅度”,是有原因的:也就是处于这种状态的进程容易被唤醒。例如说我们在上面所讲到的这个处于阻塞状态的进程,我们使用kill -9 8664向这个进程发送【9号信号】,那么这个进程就被杀死了,你也可以认为被唤醒了

在这里插入图片描述

好,接下去呢我就通过一个故事来描述一下这个【深度睡眠】到底是怎样一种状态

  • 现在在操作系统里有一个进程,它呢想要把数据写到磁盘上去,磁盘这个时候听到了它的请求,就开始写数据,不过我们知道外设尤其是像磁盘这种设备的速度都是比较慢的,所以呢这个时候进程就会一直地等待下去

在这里插入图片描述

  • 但是呢,光这么无休止地等待下去可不行。因为当前这个时候操作系统中的已经有非常多的进程了,那么OS它看不下去了,路过的时候看到当前的这个进程正处于等待状态,也不运行,于是二话不说把它终止了

在这里插入图片描述

💬 所以呢,同学们,我们要明白这么一个道理:操作系统认为当前系统能行的话就直接让用户去用就可以了,如果扛不住了就置换页表,实在不行了就去会杀进程🔪

  • 那么当这个进程被杀掉的时候,磁盘说:我在写数据的时候出了问题。它这个时候就回去找那个调度它的进程,但是发现这个进程没了???所以磁盘就特别尴尬:我要怎么办呢?写好的这个数据该不该还放在这里【一般的设备都会选择丢失】
  • 如果这份数据是普通的那还好,但如果其为银行当中那种非常重要的数据呢?该怎么办?

一场有趣的官司

那因为随着这份重要数据的丢失呢,就引发了一场官司🔨

  • 到了法庭上,法官开始问话,审讯【操作系统】、【进程】、【磁盘】这三个嫌疑人,读者认为是谁的问题的?锅在谁呢?

在这里插入图片描述

以下呢,是三个人的辩词

💬 【操作系统】:

  • 请问有没有赋予我管理它软硬件资源的权利?请问我杀掉进程是不是在特别极端的情况下杀掉的?请问我有没有履行我操作系统的职责?我的职责是保证系统不挂?数据丢失和我有什么关系?
  • 我就是在做我操作系统该做的事,如果你判我有罪了,那么请问我下次再遇到类似的情况时,到底我还做不做,我如果不杀的话,最终导致操作系统挂掉:① 数据该丢还是得丢;② 可能还会影响其他进程。这些责任谁来承担呢?

操作系统听了这番说辞后心里想了想,确实是。于是呢这个时候便把茅头指向了丢掉数据的磁盘✒ 发问道:你为什么要丢数据呢?

💬 【磁盘】:

  • 法官大人啊,这你不能怪我╮(╯▽╰)╭ 在这件事情上我就是个跑腿的,人家让我干啥就干啥,我在写入的时候就已经告诉了对方我会失败,我让他去等的,我要给它结果的,我的工作模式向来都是这样,其他磁盘也是这样
  • 如果你认为我有罪的话,是不是其他的磁盘也有问题呢?我丢失也是有原因的,其他进程也要叫我写入数据,那我也去写了,并且汇报回去了,那其他磁盘怎么没出问题呢?

操作系统一听:诶,这货说的好像确实没什么问题。那就就把视角转向了受害人即【进程】这一方。反正操作系统没错、磁盘没错,那就是你的问题喽!

💬 【进程】:

  • 法官,我可是受害人啊/(ㄒoㄒ)/~~ 我就是静静地坐在那里,“人在家中坐,锅从天上来呀”。我是被杀掉的,我怎么能有错呢?

那这个时候法官一想,它们三个说的似乎都挺有道理,难道是我错了吗?

  • 其实上面三者都没错,操作系统履行了它的职责,在关键时刻杀掉了进程;而磁盘呢则是起到了它本能的义务,丢失了没人要的数据;进程呢则因为在等候期间无意中被操作系统给杀掉了
  • 当这个进程在等磁盘写入数据这么关键的时候,应该要让其受到保护,不可以被任何人给杀掉才对

那我们在上面有提到过处于【浅度睡眠】的进程,是可以被kill掉的,那么我们就要让这个进程的状态变为【深度睡眠】才对,即[D]

  • 那既然这个进程的状态被设置为D后,当它在等待这个磁盘写入的时候,操作系统路过,看到这个进程没有跑起来空等着,于是在想把他杀掉的时候就被这块 《秒死金牌》给吓到了😮

在这里插入图片描述

  • 那么这个进程就不会被杀死了,当磁盘写完数据后告知进程,那么它就可以将自己放入【运行队列】里去运行,此时它的状态就变成R

💬 那这个时候就又有同学问了:D状态这么强大吗,那如果一个操作系统里有很多的D状态,这怎么办呢?

  • 这个同学你问得很好,确实这个D状态的话操作系统是没有办法将其杀掉的,而是要等到磁盘写入完毕或者什么事情执行完毕后其才会去自动结束这个进程,或者是在外部断电、直接拔掉电源即可
  • 一般一个系统里面如果存在D状态的话,那这个系统中的磁盘压力已经非常大了;如果存在多个D状态的话,则表示这个系统已经是非常危险了,最好马上重装一下系统💻

不过呢这个[D]就没办法在这里给读者演示了,因为D状态的进程只有处于高IO的情况才可以演示。有兴趣的可以去研究一下Linux下的命令dd

4、停止状态T

好,接下去呢我们来讲讲【停止状态T】

  • 首先我们要通过下面这句命令来查看一下对应的进程信号
kill -l
  • 此处我们要使用到的是18、19号信号

在这里插入图片描述

  • 接下去我们就来试一试如何让这个进程暂停之后又重新启动会是怎样的

在这里插入图片描述

  • 所以我们来总结一下
    • 暂停进程
    kill -19 PID
    
    • 启动进程
    kill -18 PID
    

💬 所以呢,如果我们要将一个进程给终止的话,发送19号信号即可,要让其继续启动起来,则发起18号信号

那我现在要问了,这个T停止状态和S睡眠状态有什么不同呢?

  1. stopped状态进程 完全暂停了, 其不会再接收任何信号了
  2. 一个进程通过stopped状态可以控制另一个
  3. S和D一定是在等待某种资源,而T状态可能在等待某种资源,也可能被其他进程控制

5、进程跟踪状态t

接下去呢,我们再来说说进程的跟踪状态t,还记得我们在基础篇中所学习的 GDB调试 吗?

  • 首先我们在正常状态下查看这个进程的状态,其为S睡眠状态。接下去呢我进行了【gdb】调试,在行内打上断点后,输入r运行起来后,我们再去查看这个进程的状态时,就发现其状态变成了t原因就在于这个进程在调试的时候停了下来

在这里插入图片描述

6、死亡状态X

对于【死亡状态X】来说呢,这个状态只是一个返回状态,你不会在任务列表里看到这个状态🙅‍

  • 那其实就是我们在上面所讲到的这个 杀进程
kill -9 PID

在这里插入图片描述

7、僵死状态Z —— 两个特殊进程

接下去我们来介绍一种状态叫做【僵死状态Z】,对于这个状态,我们要涉及到两个特殊的进程叫做 僵尸进程孤儿进程

① 僵尸进程

💬 首先我们要来介绍的是僵尸进程,这里呢通过一个故事来进行引入

  • 你呢,很喜欢晨跑🏃‍,这一天早晨6点又起来跑步了,当你路过一个公园的时候,遇到了一个晨练的老大爷,此时突然他就倒了下来。那你此时就非常担惊受怕了,马上拨打了110120的电话,然后守在他的身边等待救援来到,周边的人看到也都纷纷围了过来,其中不免有一些人会紧急救援。不过等了一会这个人就没了呼吸🖤

在这里插入图片描述

  • 过了十几分钟后,救护车和警车都来了,警察先封锁了,让法医过来检验一下其状况,就说:“这个人已经没救了,赶紧通知家属准备后事吧~。”

在这里插入图片描述


好,我们回归正题,来说说这个【僵尸进程】

  • 因为在救护车来之前这个人其实就已经死亡了,但是其状态还没有被检测,当前并不知道它的死因,所以我们操作系统就可能会维护当前的这个状态,这个状态即为Z状态,即[僵死状态]

💬 那我现在想问了:有一个进程暂时退出了,它要将它的状态暂时维持一段时间,问题是它维持给谁看呢?

  • 答:父进程!当一个进程退出的时候,那最关心它的便是【父进程】。因为这个父进程费了很大的劲才将这个子进程fork出来,此时呢它突然挂掉了,那么此时父进程就必须去关心一下对应的子进程退出时的原因
  • 不过光凭我们现在所学习的知识还无法去查看这个,等我们后面讲到wait()系统调用的时候再去做一定的介绍

就上面这样生冷的文字来叙述还不太行,接下去我们通过实际的案例来观察一下🔍

  1 #include <stdio.h>
  2 #include <stdlib.h>
  3 #include <unistd.h>
  4 
  5 int main(void)
  6 {
  7     pid_t id = fork();
  8     if(id == 0)
  9     {
 10         // child
 11         int cnt = 5;
 12         while(cnt)
 13         {
 14             printf("I am child, pid: %d, ppid: %d, cnt: %d\n", getpid(), getppid(), cnt);
 15             sleep(1);
 16 
 17             cnt--;
 18         }                                                                                                    
 19         exit(0);
 20     }
 21     else
 22     {
 23         // father   
 24         while(1)
 25         {
 26             printf("I am father, pid: %d, ppid: %d\n", getpid(), getppid());
 27             sleep(1);
 28         }
 29     }
 30     return 0;
 31 }
  • 运行起来可以看到,在一开始的 5s 内,父子进程还是同时在跑的,但是当子进程在循环结束后退出了。不过此时呢父进程还是运行并没有退出。所以我们在右侧查看这两个进程的状态时间就发生了一定的变化

在这里插入图片描述

  • 仔细地来看一下这个状态的变化,其中【PID】为5486,其父进程为5485,一开始它们均为S睡眠状态,但是呢到了后面子进程的状态就变成了Z僵死状态。也就意味着此时子进程已经死亡了,但是呢父进程还不知道
  • 这里还可以通过右侧的这个<defunct>这个来看,它表示【失效的、不再存在的】

在这里插入图片描述

所以我们总结一下:

💬 进程一般退出的时候,如果父进程没有主动回收子进程信息,子进程会一直让自己处于Z状态,进程的相关资源尤其是 task_struct 结构体不能被释放

那对于上面的这种子进程,我们就将其称作为是【僵尸进程】,不过呢这种进程是存在一定危害的!

  • 那一个父进程创建了很多子进程,就是不回收,是不是就会造成内存资源的浪费?是的!因为数据结构对象本身就要占用内存,想想C中定义一个结构体变量(对象),是要在内存的某个位置进行开辟空间
  • 那么对于上面的这种危害我们就称作为是【内存泄漏】,要如何去进行避免呢,之后的文章会做讲解~

② 孤儿进程

  • 上面我们讲到,当一个子进程突然退出但是父进程并没有去主动回收的话,那么此时这个子进程就会变成【僵尸进程】
  • 那看到下面我们将这个进程突然终止之后,父子进程都不见了

在这里插入图片描述
💬 那此时我想问:这个父进程突然之间退出了,但是呢它的父进程并不知晓,那为何这个父进程没有处于【僵尸状态Z】呢?

  • 因为当前这个进程是bash的子进程,当其一退出之后,bash就将其回收了。可是这个子进程为什么也没了呢?这个的话我们就要聊聊【孤儿进程】了

接下去我们来将上面测试僵尸进程的代码做个修改,让父进程先于子进程退出

  1 #include <stdio.h>
  2 #include <stdlib.h>
  3 #include <unistd.h>
  4 
  5 int main(void)
  6 {
  7     pid_t id = fork();
  8     if(id == 0)
  9     {
 10         // child
 11         int cnt = 500;
 12         while(cnt)
 13         {
 14             printf("I am child, pid: %d, ppid: %d, cnt: %d\n", getpid(), getppid(), cnt);
 15             sleep(1);
 16 
 17             cnt--;
 18         }
 19         exit(0);
 20     }
 21     else
 22     {
 23         int cnt = 5;
 24         // father   
 25         while(cnt--)                                                                                
 26         {
 27             printf("I am father, pid: %d, ppid: %d\n", getpid(), getppid());
 28             sleep(1);
 29         }
 30     }
 31     return 0;
 32 }
  • 然后通过进程状态追踪来看看,我们观察到一开始两个父进程和子进程是同步在走的,但是呢过了一会父进程终止了,只有子进程还在跑,此时我们需要留意的不是其进程状态,而是这个子进程的PPID即父进程的PID,它由原先的【19956】变成了【1】,这是为什么呢?

在这里插入图片描述

  • 这里的话就要给读者普及一下了,对于PID的值为1的进程,我们一般将其称作为是【系统进程】。我们可以使用下面这句指令去查找一下
ps ajx | head -1 && ps ajx | grep systemd
  • 然后我们看到这个【系统进程】的PID即为1

在这里插入图片描述

那为何会出现上面这种现象呢?是这个子进程突然换父进程了吗?

  • 对于父子进程来说,如果父进程先于子进程结束的话,那么这个子进程就会被称作为是【孤儿进程】,对于孤儿进程来说呢,它会有1号进程即【系统进程】所领养,因为此时这个进程没有了父进程了,所以需要有人去带领它

💬 但是这个孤儿进程为什么要被领养呢?

  • 其实很简单,既然它是一个进程的话,最终都是要退出的,但是没了父亲的话就只能由系统进程来进行维护了

那其实我们就可以解释上面的事情了

  • 当这个父进程退出的时候,它是被bash给回收了,那么对于父进程的子进程来说,我们刚才谈到了,如果一个进程的父进程先于子进程结束的话,那么在这一刻这个子进程立马变成了【孤儿进程】,被系统领养并回收了,所以我们就看不到了

在这里插入图片描述

四、总结与提炼

最后来总结一下本文所学习的内容📖

  • 在本文中,我们主要讲解了两个大点,一个是在操作系统中所常见的一些进程状态,它们分别为:运行状态、阻塞状态、挂起状态
    • 所被调度的、处于运行队列里的这些进程所处的状态我们称之为 运行状态R
    • 处于等待队列中,同时在等待外设相应的进程所处的状态我们称之为 阻塞状态R
    • 当把一个进程的 数据和代码都重新交换到外设当中,进程所处的状态我们称之为 挂起状态R
  • 接下去我们又介绍了在Linux系统下的七种进程状态,分别是:运行状态R、浅度睡眠状态S、深度睡眠状态D、停止状态T、进程跟踪状态t、死亡状态X、僵死状态Z
    • 当一个进程没有在等待IO流的时候,其就会处于 运行状态R
    • 使用sleep()函数可以很好地使一个进程处于浅度睡眠状态S
    • 如果不想让一个进程在等待磁盘操作的时候被操作系统杀掉,则可让其处于 深度睡眠状态D
    • 可以向一个进程发送【19】号信号使其暂停,处于停止状态T继续发送【18】号信号的话则可以使其重新启动
    • 在【gdb】的环境下去运行一个断点的话则可以使其处于进程跟踪状态t
    • 使用kill -9 PID就可以杀掉一个进程,使其处于死亡状态X
    • 如果让一个子进程在父进程不知晓的时候退出可,那么其就会处于僵死状态Z,变为【僵尸进程】;若是在父子进程中父进程先于子进程退出的话,那么这个子进程就会变成【孤儿进程】

以上就是本文要介绍的所有内容,感谢您的阅读🌹

在这里插入图片描述

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

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

相关文章

算法竞赛备赛之搜索与图论训练提升,暑期集训营培训

目录 1.DFS和BFS 1.1.DFS深度优先搜索 1.2.BFS广度优先搜索 2.树与图的遍历&#xff1a;拓扑排序 3.最短路 3.1.迪杰斯特拉算法 3.2.贝尔曼算法 3.3.SPFA算法 3.4.多源汇最短路Floy算法 4.最小生成树 4.1.普利姆算法 4.2.克鲁斯卡尔算法 5.二分图&#xff1a;染色法…

嵌入式学习之strcpy、memset、realloc、malloc使用方法

今天主要针对C语言的strcpy memset realloc mallooc函数进行了学习。 char* strcpy(char* destination,const char* source); void memset ( void *s , char ch, unsigned n )&#xff1b; void* realloc(void* memblock, size_t size)&#xff1b; void *malloc(size_t si…

tkinter打造三维绘图系统,附源代码

文章目录 输入数据加载数据绘图函数源代码 Python绘图系统系列&#xff1a;将matplotlib嵌入到tkinter 简单的绘图系统 数据导入 输入数据 三维绘图需要一个新的坐标变量&#xff0c;设置为z&#xff0c;这个改改UI就可以办到&#xff0c;并不困难。但是&#xff0c;此前用于…

git安装介绍

一、分布式版本控制系统Git概述 1.1 分布式版本控制系统Git介绍 版本控制定义 记录和跟踪项目中各文件内容的改动变化 保存项目的版本历史&#xff0c;以及改动原因&#xff0c;从而让用户能够查看各个历史版本 版本控制系统也是帮助人员进行协作开发的利器 为什么需要版本…

WebRTC音视频通话-WebRTC本地视频通话使用ossrs服务搭建

iOS开发-ossrs服务WebRTC本地视频通话服务搭建 之前开发中使用到了ossrs&#xff0c;这里记录一下ossrs支持的WebRTC本地服务搭建。 一、ossrs是什么&#xff1f; ossrs是什么呢&#xff1f; SRS(Simple Realtime Server)是一个简单高效的实时视频服务器&#xff0c;支持RTM…

福康源:用孝道温暖每一个心灵,共筑幸福健康新人生!

福康源&#xff1a;用孝道温暖每一个心灵&#xff0c;共筑幸福健康新人生 孝道的光芒&#xff1a;福康源的初心 在浮躁的现代社会&#xff0c;孝道的力量正被越来越多的人忽略。然而&#xff0c;福康源的初心却始终坚守孝顺的真谛。孝道不仅是对父母的敬爱&#xff0c;更是一种…

【解析postman工具的使用---基础篇】

postman前端请求详解 主界面1.常见类型的接口请求1.1 查询参数的接口请求1.1.1 什么是查询参数?1.1.2 postman如何请求 1.2 ❤表单类型的接口请求1.2.1 复习下http请求1.2.2❤ 什么是表单 1.3 上传文件的表单请求1.4❤ json类型的接口请求 2. 响应接口数据分析2.1 postman的响…

程序设计 树基础

✅作者简介&#xff1a;人工智能专业本科在读&#xff0c;喜欢计算机与编程&#xff0c;写博客记录自己的学习历程。 &#x1f34e;个人主页&#xff1a;小嗷犬的个人主页 &#x1f34a;个人网站&#xff1a;小嗷犬的技术小站 &#x1f96d;个人信条&#xff1a;为天地立心&…

Lorilla LLM - 面向API调用生成的专用AI大模型

Gorilla 是一种先进的大型语言模型 (LLM)&#xff0c;旨在与各种 API 有效交互&#xff0c;从而增强 LLM 在实际应用中的功能。 Gorilla LLM的相关链接&#xff1a;官网 | github | 论文。 推荐&#xff1a;用 NSDT编辑器 快速搭建可编程3D场景 1、Gorilla LLM简介 通过使用自…

CentOS 项目作出声明,宣称自家 Linux 社区 “始终向所有人开放”

导读在红帽 RHEL 开源事件后&#xff0c;许多兼容 RHEL 的发行版最近都进行了表态&#xff0c;CentOS 项目也在日前作出了声明&#xff0c;宣称自家社区 “始终向所有人开放”。 据悉&#xff0c;CentOS 项目董事会日前在官方博客发布了一则公告&#xff0c;内容主要涉及“ Ce…

拒绝摆烂!C语言练习打卡第一天

&#x1f525;博客主页&#xff1a;小王又困了 &#x1f4da;系列专栏&#xff1a;每日一练 &#x1f31f;人之为学&#xff0c;不日近则日退 ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ &#x1f5d2;️前言&#xff1a; 在前面我们学习完C语言的所以知识&#xff0c;当…

LangChain手记 Models,Prompts and Parsers

整理并翻译自DeepLearning.AILangChain的官方课程&#xff1a;Models,Prompts and Parsers 模型&#xff0c;提示词和解析器&#xff08;Models, Prompts and Parsers&#xff09; 模型&#xff1a;大语言模型提示词&#xff1a;构建传递给模型的输入的方式解析器&#xff1a;…

NAS搭建指南二——影视资源中心

1. 刮削 这一部分是利用 tinyMediaManager 进行影视资源的评分、简介以及图片的搜集工作tinyMediaManager 官方网站地址下载及安装过程&#xff1a;略我的主要修改的设置项如下所示&#xff1a; 使用方法&#xff1a; a. 点击更新媒体库 b. 选择影片–>右键单击–>…

Ubuntu 20.04(服务器版)安装 Anaconda

0、Anaconda介绍 Anaconda是一个开源的Python发行版本&#xff0c;包含了包括Python、Conda、科学计算库等180多个科学包及其依赖项。因此&#xff0c;安装了Anaconda就不用再单独安装CUDA、Python等。 CUDA&#xff0c;在进行深度学习的时候&#xff0c;需要用到GPU&#xf…

生信分析pandas数据处理 Python简明教程 | 视频18

开源生信 Python教程 生信专用简明 Python 文字和视频教程 源码在&#xff1a;https://github.com/Tong-Chen/Bioinfo_course_python 目录 背景介绍 编程开篇为什么学习Python如何安装Python如何运行Python命令和脚本使用什么编辑器写Python脚本Python程序事例Python基本语法 数…

软件开发中常用数据结构介绍:C语言链表

工作之余来写写C语言相关知识&#xff0c;以免忘记。今天就来聊聊C语言链表&#xff0c;我是分享人M哥&#xff0c;目前从事车载控制器的软件开发及测试工作。 学习过程中如有任何疑问&#xff0c;可底下评论&#xff01; 如果觉得文章内容在工作学习中有帮助到你&#xff0c;麻…

干不完根本干不完,我也不想加班,快来围观时间管理大师

时间不够用&#xff0c;怎么办&#xff1f; 成功不靠加班。生产队的驴都不加班&#xff0c;你加什么班&#xff1f;到点就下班&#xff0c;该玩玩&#xff0c;该学习认真学&#xff0c;累了就睡觉。 你可以做任何事&#xff0c;但不必做所有事。 时间管理&#xff0c;不是管…

(十七)大数据实战——Hive的hiveserver2服务安装部署

前言 HiveServer2 是 Apache Hive 的一个服务器端组件&#xff0c;用于支持客户端与 Hive 进行交互和执行查询。HiveServer2服务的作用是提供jdbc/odbc接口&#xff0c;为用户提供远程访问Hive数据的功能。HiveServer2 允许多个客户端同时连接并与 Hive 交互。这些客户端可以通…

HOT91-不同路径

leetcode原题链接&#xff1a;不同路径 题目描述 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish” &#xff09;…

shell编程免交互

免交互 定义&#xff1a; 一键执行&#xff0c;无需人工干预&#xff0c;不需要人为控制&#xff0c;就可以完成的操作&#xff0c;继续自动化运维 免交互在linux中的运用&#xff1a; 对于Linux操作系统中&#xff0c;有许多操作都会触及到交互&#xff08;根据系统的指示做…