进程状态(R|S|D|t|T|X|Z)、僵尸进程及孤儿进程

文章目录

  • 一.进程状态
    • 进程排队
    • 状态:运行、阻塞、挂起
  • 二.Linux下的进程状态
    • R 运行状态(running)
    • S 睡眠状态(sleeping)
    • D 磁盘休眠状态(Disk sleep)
    • t 停止、暂停状态(tracing stopped)
    • T 停止、暂停状态(stopped)
    • X 死亡状态(dead)
  • 三.僵尸进程 Z(zombie)
  • 四.孤儿进程

一.进程状态

操作系统的基本职责是控制进程的执行,这包括确定交替执行的方式和给进程分配资源。在设计控制进程的程序中,第一步就是描述进程所表现出的行为。

一般来说,有三种进程状态模型:三状态模型、五状态模型和七状态模型。
在这里插入图片描述

每个操作系统的具体实现都是在这些模型的基础上进行设计的,其中最核心的就是三状态模型。五状态模型在三状态模型的基础上添加了创建态和结束态。七状态模型在五状态模型的基础上添加了挂起状态。

运行态:进程占有CPU,并在CPU上运行
就绪态:一个进程已经具备运行条件,但没有分配CPU,暂时不能运行。当调度到CPU时可以立刻运行。
等待态:(阻塞态、封锁态、睡眠态),进程因某个事件发生而不能立即运行。即使CPU空闲,该进程也不可运行。

进程排队

一个进程,自创建之后不是一直在CPU上面运行的,操作系统会基于时间片来轮流执行进程,以保证每个进程都能运行。单CPU下,同一时间只能一个进程运行,那么必然会出现进程排队,出现排队一定是在等待某种资源,可能是时间片,也可能是软硬件资源。
只要排队,肯定是进程的PCB即task_struct在排队,那么就要将其组织到队列中。实际上,task_struct可以被连入多个数据结构中:双链表,队列等等,不同以往,若仅想将task_struct组织到一个数据结构例如链表中,只用直接将其封装起来:

struct list{task_struct *_next;task_struct *_prev;
}

但要将其放入多个数据结构中,仍这样组织,操作系统不仅要维护PCB,还要维护各种链表队列,有些冗杂。在Linux中,直接将链表结点、队列结点封装到task_struct中间,这样操作系统对进程的管理就只会对PCB进行管理,不会涉及其他的数据结构。

struct task_struct {//...//指向运行队列的指针struct list_head run_list;  //用于将系统中所有的进程连成一个双向循环链表, 其根是init_taskstruct task_struct *next_task, *prev_task;//...
};

每个CPU会维护一个运行队列,当访到某一进程的run_list结点时,这结点是整个PCB的一个成员,那么怎么问其他数据呢❓

拿一个简单的结构体来说:

struct S{int a;int b;int c;
};

知道&c怎么得到&S呢?&c = &S + 偏移量,结构体成员越靠后,那么偏移量就越大,同时取地址取的是低地址,那么只要根据首地址和偏移量就可以得到所有成员的数据了。小到int大到struct都是这样的。

在Linux内核中是这样实现的:&n - &((task_struct*)0->n),这样就能得到task_struct对象的首地址了。

状态:运行、阻塞、挂起

所谓状态,本质就是一个task_struct中的整型变量,不同状态对应不同的整数值。

//说明了该进程是否可以执行,还是可中断等信息
volatile long state; 

状态决定了进程的后续动作

  • 运行态
    一个CPU管理一个运行队列,只要进程在运行队列排队,那么其都在运行状态。
    在这里插入图片描述

  • 阻塞态
    操作系统对硬件管理也是先描述再组织,每个硬件也有其数据结构。当进程需要某种资源例如硬件资源时,但是资源没有就绪,则会先出运行队列,随后进入该硬件对象中的等待队列,此时该进程进入阻塞状态,即使CPU空闲,也不会进入CPU的运行队列。

  • 挂起态
    挂起不常见,这里谈一下阻塞挂起。
    前提是计算机资源吃紧,磁盘中有一块SWAP分区,当内存吃紧,由于操作系统为了提供安全稳定的环境,这时会将阻塞进程对应的代码数据唤出到磁盘中的SWAP分区,为了之后的唤入做准备。当内存资源不再吃紧,会将代码数据再唤入内存中。这里的唤入唤出,不会对PCB进行交换。
    在这里插入图片描述

一般来说,SWAP分区不要超过内存大小,由于访问外设的速度慢,唤入唤出就是用效率换稳定,若SWAP分区太大,系统I/O频率会高,操作系统效率会变低。

二.Linux下的进程状态

为了方便查看进程状态,使用以下指令来循环ps ajx

while : ; do ps axj | head -1 && ps axj | grep myprocess | grep -v grep; sleep 1; done 

下面的状态在kernel源代码里定义:

/*
* The task state array is astrange "bitmap" of
* reasons to sleep. Thus"running" is zero, and
* you can test for combinationsof others with
* simple bit tests.
*/
static const char * consttask_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 */
};

Linux 内核中的进程状态是一个指针数组static const char * consttask_state_array[]

R 运行状态(running)

并不意味着进程一定在运行中,它表明进程要么是在运行中要么在运行队列里。

在Linux中没有就绪态,将其也视为运行态。

下面运行下面一个程序让其变为进程,并通过指令查看该进程的状态:

#include <stdio.h>
#include <unistd.h>int main()
{while (1) {printf("Hello!\n");sleep(1);}return 0;
}

该进程一直在循环,其进程状态应该是R,但是结果却与猜想不一致,发现其状态是S(睡眠状态)。

  PPID     PID    PGID     SID TTY        TPGID STAT   UID   TIME COMMAND
343984  345559  345559  343984pts/0     345559 S+       0  0:00 ./myprocess

这是因为这段代码中有printf,打印到显示器上显然需要访问外设,这时操作系统会将该进程状态改为阻塞状态并放入显示器的等待队列中,等待相比打印是漫长的,很难抓取到该进程状态为R的时刻。这时只留一个死循环,再运行就可以了。

int main()
{while (1) {}return 0;
}

由于该程序不会进行任何I/O操作,自然其一直在运行队列。这里的R就代表该进程一直在运行状态,+表示其为前台进程,可以通过ctrl+C来结束进程;若没有则表示后台进程,那么就不可以了。

PPID     PID    PGID     SIDTTY        TPGID STAT   UID  TIME COMMAND
343984  346537  346537  343984pts/0     346537 R+       0  0:05 ./myprocess

S 睡眠状态(sleeping)

睡眠状态对应操作系统概念中的阻塞态,意味着进程在进程在等待事件完成。
S为可中断睡眠|浅度睡眠

除了上面进程因等待外设而进入睡眠状态,也可以使用sleep函数让进程睡眠。

#include <stdio.h>
#include <unistd.h>int main()
{while (1){sleep(1);}return 0;
}

查看该进程状态为S+

PPID     PID    PGID     SIDTTY        TPGID STAT   UID  TIME COMMAND
343984  347903  347903  343984pts/0     347903 S+       0  0:00 ./myprocess

由于其为可中断睡眠,可以通过kill -19将其唤醒

PPID     PID    PGID     SID TTY        TPGID STAT   UID   TIME COMMAND
343984  348168  348168  343984pts/0     348168 S+       0  0:00 ./myprocessPPID     PID    PGID     SID TTY        TPGID STAT   UID   TIME COMMAND
343984  348168  348168  343984pts/0     343984 T        0   0:00 ./myprocess

D 磁盘休眠状态(Disk sleep)

对应阻塞态。也叫不可中断睡眠状态,在这个状态的进程通常会等待IO的结束。

通常来说进程的睡眠状态都为浅度睡眠状态,但当一个进程向磁盘写入大量数据时,写入过程很漫长,进程需要一直处在阻塞态,如果这个过程被打断,很容易出现磁盘数据和内存数据不一致的情况。操作系统当内存严重吃紧的时候,会终止一些在阻塞态过长的进程,但其实这个时候该进程还在等待磁盘写入的结果,这种状态肯定是不稳定的。

为了避免这种情况,当一个进程在向磁盘进行读写时,为了保证内存数据和磁盘数据一致,在接收到磁盘读写结果之前,该进程一直会处在D状态,不可以被打断。

若进程处于D状态,即使使用kill -9也无法将该进程终止。

t 停止、暂停状态(tracing stopped)

对应阻塞态,该状态也被称为调试状态。

当调试一个程序时,在断点处进程暂停,此进程为t状态

 PPID     PID    PGID     SID TTY        TPGID STAT   UID   TIME COMMAND662865  662893  662893  661630 pts/0     662865 t      0   0:00 /root/learning/Linux-learning/Process/myprocess

T 停止、暂停状态(stopped)

对应阻塞态,当进程有危险操作时会被暂停。

可以通过发送19信号,让进程暂停:kill -19 (进程PID)

PPID     PID    PGID     SIDTTY        TPGID STAT   UID   TIMECOMMAND
661630  663368  663368  661630 pts0     661630 T        0   0:00 .myprocess

X 死亡状态(dead)

对应结束态,当一个进程结束时产生的状态。这个状态只是一个返回状态,不会在任务列表里看到这个状态。

三.僵尸进程 Z(zombie)

对应结束态,X状态之前的状态

进程创建出来完成任务后,不会立即死亡即为X状态,而是会先变成Z状态,若其父进程不进行回收,则会一直处在Z状态。
这是由于父进程需要知道子进程执行的情况,此时,该进程的代码和数据会被直接释放,但保留PCB在内存中供父进程读取子进程执行情况,例如exit_state,exit_code,exit_signal

下面模拟僵尸进程,当子进程结束,父进程不退出也不读取子进程的执行情况即不进行回收,此时子进程将一直保持Z状态:

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <stdlib.h>int main()
{pid_t id = fork();if (id == 0){ // 子进程一共会存在5秒钟int cnt = 5;while (cnt--){printf("I'm child process. \n");sleep(1);}printf("I'm child process,  zombie  to  be checked.\n");exit(0); // 子进程直接退出}else {// 父进程不结束也不回收子进程while (1){sleep(1);}}return 0;
}
root@hcss-ecs-e6eb:~/learning/Linux-learning/Process# ./myprocess 
I'm child process.
I'm child process.
I'm child process.
I'm child process.
I'm child process.
I'm child process, zombie to be checked.
661630  669793  669793  661630 pts0     669793 S+       0   0:00 .myprocess
669793  669794  669793  661630 pts0     669793 Z+       0   0:00[myprocess] <defunct>

此时父进程不读取且不退出,那么子进程的PCB就会一直在内存中,造成内存泄漏。

四.孤儿进程

孤儿进程就是其父进程推出了,该进程仍然存在。

下面让父进程3秒后就退出,观察细节

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <stdlib.h>int main()
{pid_t id = fork();if (id == 0){// 子进程不退出while (1){printf("I'm child process,     PID:%d, never stop.\n",  getpid ());sleep(1);}}else {// 父进程3秒后退出int n = 3;while (n){printf("I'm father process,    PID:%d, %d left.\n", getpid    (), n--);sleep(1);}exit(0);}return 0;
}

下面是父进程退出前后的表现:

I'm father process, PID:677010, 3 left.
I'm child process, PID:677011, never stop.
I'm father process, PID:677010, 2 left.
I'm child process, PID:677011, never stop.
I'm father process, PID:677010, 1 left.
I'm child process, PID:677011, never stop.
I'm child process, PID:677011, never stop.
   PPID     PID    PGID     SID TTY        TPGID STAT   UID   TIME COMMAND676915  677010  677010  676915 pts/0     677010 S+       0   0:00 ./myprocess677010  677011  677010  676915 pts/0     677010 S+       0   0:00 ./myprocessPPID     PID    PGID     SID TTY        TPGID STAT   UID   TIME COMMAND1  677011  677010  676915 pts/0     676915 S        0   0:00 ./myprocessPPID     PID    PGID     SID TTY        TPGID STAT   UID   TIME COMMAND1  677011  677010  676915 pts/0     676915 S        0   0:00 ./myprocess

子进程的ppid直接变为1,通过指令查看该进程为:

PPID     PID    PGID     SIDTTY        TPGID STAT   UID   TIMECOMMAND0       1       1       1 ?             -1 Ss       0   1:18 /lib/systemd/systemd --system --deserialize 41 noibrs

这个1号进程,就是Linux启动后,第一被创建的用户进程。把1号进程领养的程,称之为孤儿进程,此时只有发9信号或者系统关机才可以使该进程结束。

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

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

相关文章

为什么要将PDF转换为CSV?CSV是Excel吗?

在企业和数据管理的日常工作中&#xff0c;PDF文件和CSV文件承担着各自的任务。PDF通常用于传输和展示静态的文档&#xff0c;而CSV因其简洁、易操作的特性&#xff0c;广泛应用于数据存储和交换。如果需要从PDF中提取、分析或处理数据&#xff0c;转换为CSV格式可能是一个高效…

Starlink卫星动力学系统仿真建模第十讲-基于SMC和四元数的卫星姿态控制示例及Python实现

基于四元数与滑模控制的卫星姿态控制 一、基本原理 1. 四元数姿态表示 四元数运动学方程&#xff1a; 3. 滑模控制设计 二、代码实现&#xff08;Python&#xff09; 1. 四元数运算工具 import numpy as npdef quat_mult(q1, q2):"""四元数乘法""…

CSS—引入方式、选择器、复合选择器、文字控制属性、CSS特性

目录 CSS 1.引入方式 2.选择器 3.复合选择器 4.文字控制属性 5.CSS特性 CSS 层叠样式表&#xff0c;是一种样式表语言&#xff0c;用来描述HTML文档的呈现 书写时一般按照顺序&#xff1a;盒子模型属性—>文字样式—>圆角、阴影等修饰属性 1.引入方式 引入方式方…

OpenHarmony-4.基于dayu800 GPIO 实践(2)

基于dayu800 GPIO 进行开发 1.DAYU800开发板硬件接口 LicheePi 4A 板载 2x10pin 插针&#xff0c;其中有 16 个原生 IO&#xff0c;包括 6 个普通 IO&#xff0c;3 对串口&#xff0c;一个 SPI。TH1520 SOC 具有4个GPIO bank&#xff0c;每个bank最大有32个IO&#xff1a;  …

win11 24h2 远程桌面 频繁断开 已失去连接 2025

一、现象 Windows11自升级2025年2月补丁后版本号为系统版本是26100.3194&#xff0c;远程桌面频繁断开连接&#xff0c;尝试连接&#xff0c;尤其在连接旧的server2012 二、临时解决方案 目前经测试&#xff0c;在组策略中&#xff0c;远程桌面连接客户端&#xff0c;关闭客户…

rust学习笔记6-数组练习704. 二分查找

上次说到rust所有权看看它和其他语言比有什么优势&#xff0c;就以python为例 # Python3 def test():a [1, 3, -4, 7, 9]print(a[4])b a # 所有权没有发生转移del b[4]print(a[4]) # 由于b做了删除&#xff0c;导致a再度访问报数组越界if __name__ __main__:test() 运行结…

Windows安装NVIDIA显卡CUDAD调用GPU,适用于部署deepseek r1

显卡、显卡驱动、CUDA之间的关系 显卡&#xff1a;&#xff08;GPU&#xff09;&#xff0c;主流是NVIDIA的GPU&#xff0c;因为深度学习本身需要大量计算。GPU的并行计算能力&#xff0c;在过去几年里恰当地满足了深度学习的需求。AMD的GPU基本没有什么支持&#xff0c;可以不…

基于无人机遥感的烟株提取和计数研究

一.研究的背景、目的和意义 1.研究背景及意义 烟草作为我国重要的经济作物之一&#xff0c;其种植面积和产量的准确统计对于烟草产业的发展和管理至关重要。传统的人工烟株计数方法存在效率低、误差大、难以覆盖大面积烟田等问题&#xff0c;已无法满足现代烟草种植管理的需求…

《深度学习实战》第3集:循环神经网络(RNN)与序列建模

第3集&#xff1a;循环神经网络&#xff08;RNN&#xff09;与序列建模 引言 在深度学习领域&#xff0c;处理序列数据&#xff08;如文本、语音、时间序列等&#xff09;是一个重要的研究方向。传统的全连接网络和卷积神经网络&#xff08;CNN&#xff09;难以直接捕捉序列中…

【前沿探索篇七】【DeepSeek自动驾驶:端到端决策网络】

第一章 自动驾驶的"感官革命":多模态神经交响乐团 1.1 传感器矩阵的量子纠缠 我们把8路摄像头+4D毫米波雷达+128线激光雷达的融合称为"传感器交响乐",其数据融合公式可以简化为: def sensor_fusion(cam, radar, lidar):# 像素级特征提取 (ResNet-152…

可狱可囚的爬虫系列课程 13:Requests使用代理IP

一、什么是代理 IP 代理 IP&#xff08;Proxy IP&#xff09;是一个充当“中间人”的服务器IP地址&#xff0c;用于代替用户设备&#xff08;如电脑、手机等&#xff09;直接与目标网站或服务通信。用户通过代理IP访问互联网时&#xff0c;目标网站看到的是代理服务器的IP地址&…

https:原理

目录 1.数据的加密 1.1对称加密 1.2非对称加密 2.数据指纹 2.1数据指纹实际的应用 3.数据加密的方式 3.1只使用对称加密 3.2只使用非对称加密 3.3双方都使用对称加密 3.4非对称加密和对称加密一起使用 4.中间人攻击 5.CA证书 5.1什么是CA证书 CA证书的验证 6.https的原理 1.数据…

Github项目管理之 其余分支同步main分支

文章目录 方法&#xff1a;通过 Pull Request 同步分支1. **创建一个从 main 到目标分支的 Pull Request**2. **合并 Pull Request** 注意事项总结 在 GitHub 网页上&#xff0c;你可以通过 Pull Request 的方式将一个分支&#xff08;例如 main 分支&#xff09;的修改同步到…

Aseprite绘画流程案例(5)——花盆

1.最终图片效果 参考素材来源于&#xff1a;手绘像素画第三课&#xff1a;像素画盆花示范&#xff08;无参考图&#xff09;_哔哩哔哩_bilibili 2.流程 1.新建画布40X27的画布&#xff0c;打开显示网格&#xff0c;背景色为白色 2.画出梯形的盆 3.给盆进行亮暗对比上色 4.添…

【模板】csdn markdown语法演示

这里写自定义目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个…

【Python系列】PYTHONUNBUFFERED=1的作用

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

Adobe After Effects的动画制作

作者&#xff1a;余佳琪 目录 一、 前言 二、 可动骨骼的选择 三、 运动曲线的设置 四、 图层的选定与应用 五、 插件的应用&#xff08;阴影&#xff0c;高光&#xff0c;特效&#xff09; 六、 导出 一、 前言 在当今世界&#x…

可狱可囚的爬虫系列课程 14:10 秒钟编写一个 requests 爬虫

一、前言 当重复性的工作频繁发生时&#xff0c;各种奇奇怪怪提高效率的想法就开始萌芽了。当重复代码的模块化封装已经不能满足要求的时候&#xff0c;更高效的方式就被揭开了神秘的面纱。本文基于这样的想法&#xff0c;来和大家探讨如何 10 秒钟编写一个 requests 爬虫程序。…

QNX上如何抓tracelogger日志

背景 因QNX侧 QVM的分析CPU负载问题在android侧使用trace无法分析&#xff0c;故QNX侧的CPU负载问题需要用到tracelogger日志分析。 例如&#xff1a;使用hogs -l 42|grep qvm 中发现qvm的cpu负载 30%多 但是使用trace日志在Perfetto又查不到qvm信息&#xff0c;则需要抓取qn…

DeepSeek开源周 Day02:从DeepEP开源趋势重新审视大模型Infra

DeepEP 今天DeepSeek开源周第二天&#xff0c;开放了DeepEP仓库&#xff0c;属实看了下源码&#xff0c;和昨天FlashMLA一样&#xff0c;C权重&#xff08;包括CUDA&#xff09;还是占据了绝对部分&#xff0c;作为调包侠的我&#xff0c;看到之后望而却步&#xff0c;想看原理…