欢迎来到Cefler的博客
博客主页:那个传说中的man的主页
个人专栏:题目解析
推荐文章:题目大解析3
进程状态是指一个进程在其生命周期中可能处于的不同状态。常见的进程状态包括以下几种:
新建状态(New):当一个新的进程被创建时,它处于新建状态。此时操作系统已经为该进程分配了必要的资源,但还没有开始执行。
就绪状态(Ready):当一个进程已经准备好执行,但由于CPU资源有限,当前没有可用的CPU时间片来运行它时,该进程处于就绪状态。就绪状态的进程将等待分配到CPU资源。
运行状态(Running):当一个进程获得CPU时间片并正在执行时,它处于运行状态。在单核处理器情况下,一次只能有一个进程处于运行状态;而在多核处理器或多任务操作系统中,可以有多个进程同时处于运行状态。
阻塞状态(Blocked):当一个进程无法继续执行,因为它需要等待某些事件发生(如I/O操作完成、等待资源释放等),此时它将进入阻塞状态。在阻塞状态下,该进程会被移出CPU,并暂时停止执行,直到事件发生并且进程可以继续执行。
结束状态(Terminated):当一个进程完成它的任务或被操作系统终止时,进程进入结束状态。在结束状态下,进程的所有资源将被释放,并等待操作系统回收。
一个进程可以按照以上状态之间的转换进行切换。例如,当一个新建进程变为就绪状态时,它等待CPU分配资源;当分配到CPU资源后,进程从就绪状态变为运行状态;如果进程需要等待某些事件完成,它将从运行状态转为阻塞状态;一旦事件发生,进程从阻塞状态恢复并重新进入就绪状态等待CPU时间片,最终在任务完成时进入结束状态。
操作系统通过对进程状态的管理,可以合理调度和执行进程,实现多任务的并发执行。了解进程状态的转换可以帮助我们理解进程的行为和调度机制,以及进行进程调优和排查问题时的分析。
在代码的角度上
进程状态,就是描述结构体PCB中的一个字段,就是PCB中的一个变量,int status
所谓的状态变化,本质就是修改整型变量
在操作系统的角度上
只要在运行队列中的进程,状态都是运行状态。
每一个cpu都会在系统层面维护一个运行队列
我们的代码中,一定或多或少会访问系统中的某些资源
比如硬件资源:磁盘,键盘,网卡等……
比如访问键盘,但是用户始终不在键盘上进行输入,也就是进程要访问的资源未就绪 ,
所以进程代码就无法执行后续的代码
资源未就绪,操作系统要不要知道?
答:
必须的,因为操作系统要管理硬件,提到管理,就要先描述再组织。
OS是最先知道它所管理设备的状态变化
所以硬件资源也有属于它的"PCB",即描述结构体
struct dev
{
int type;
int status;
struct dev* next;
PCB* wait_queue;//若被阻塞,就放入这个队列中等待
//更多属性
}
可以表示为如下这张图
所以总结一下,进程状态变化的本质!!:
1.更改pcb中的status变量
2.将pcb连入不同的队列(运行/等待)
我们所说的所有的进程,只和pcb有关,与代码本身无关
所以当我们看到进程堵塞了,
现象是进程卡住了,但内核原因是pcb没有在运行队列中&&状态不是running,cpu不调度你的进程
挂起状态(Suspended)是进程可能处于的一种特殊状态,与其他常见的进程状态(新建、就绪、运行、阻塞、结束)不同。当一个进程被挂起时,它的执行会被临时中断,并且实际占用的系统资源会被释放,以便其他进程能够继续运行。
挂起状态通常分为两种类型:可恢复挂起和不可恢复挂起。
可恢复挂起(可中断挂起):当一个进程被可恢复挂起时,它的执行可以被恢复。这种状态下,进程被暂停执行,但其内部状态仍然保存在内存中,以便稍后重新开始执行。常见的可恢复挂起情况包括通过按下 Ctrl+Z 键将前台进程切换到后台、通过操作系统的挂起命令将进程挂起等。
不可恢复挂起(不可中断挂起):当一个进程被不可恢复挂起时,它的执行无法被恢复。这种状态下,进程的执行被完全中断,其内部状态不再保存。常见的不可恢复挂起情况包括进程被操作系统强制终止、进程因错误或异常而被终止等。
进程可以被挂起的原因包括但不限于以下情况:
当满足特定条件时,操作系统可以恢复可恢复挂起的进程的执行,并将其状态切换为就绪状态,以便进一步执行。在恢复执行之前,操作系统可能会将挂起的进程从磁盘交换到内存中。
总而言之,挂起状态是指进程执行被临时中断并释放系统资源的特殊状态。可恢复挂起的进程可以在稍后恢复执行,而不可恢复挂起的进程则无法继续执行。挂起状态的使用可以帮助操作系统合理分配资源和调度进程,以提高系统的效率和响应性。
在Linux操作系统下,进程状态可以分为以下几种:
运行状态(Running):表示进程正在运行或占用CPU执行指令。
就绪状态(Ready):表示进程已经准备好执行但尚未获得CPU资源。就绪状态的进程等待调度器将其分配给可用的CPU。
阻塞状态(Blocked):表示进程由于某种原因无法继续执行,需要等待某个事件发生或满足某个条件才能继续执行。常见的阻塞事件包括等待I/O操作完成、等待信号量、等待互斥锁等。
僵死状态(Zombie):当一个进程的执行已经结束,但其父进程尚未通过wait()系统调用来获取其终止状态时,该进程就会成为僵死进程,也称为僵尸进程。僵尸进程仍然在进程表中保留一些必要的信息,等待父进程处理,并且不再占用系统资源。
停止状态(Stopped):表示进程被暂时停止并且不会继续执行。进程可以通过信号(如SIGSTOP、SIGTSTP)或调试器(如gdb)中断来进入停止状态。与阻塞状态不同,停止状态下的进程不会被调度器重新唤醒。
前台状态(Foreground):表示与终端交互的进程当前在前台执行。通常情况下,用户可以通过终端输入来与前台进程进行交互,而其他非前台进程处于后台执行。
后台状态(Background):表示进程在后台执行,没有与终端进行交互。后台进程不会接收终端输入,但可以在后台继续执行。
进程状态之间的转换是动态的,并且由操作系统内核和调度器负责管理。例如,当一个进程变为就绪状态时,它等待调度器将其从就绪队列中选中并切换到运行状态;当进程遇到某个条件并需要等待时,它会从运行状态切换到阻塞状态等待事件发生。
了解Linux中的进程状态对于调试、性能优化和系统监控都非常重要,可以帮助我们理解进程的行为、调度机制以及进程间的交互。
在Linux中,可以通过ps
命令或者top
命令查看进程状态。以下是常见的Linux中用于标识进程状态的符号:
并不意味着进程一定在运行中,它表明进程要么是在运行中要么在运行队列
里
S:表示睡眠状态(Sleeping),即进程正在等待某个事件的发生。
D:表示不可中断的睡眠状态(Uninterruptible Sleep),进程正在等待一些无法中断的事件的发生。
T:表示停止状态(Stopped),即进程被暂停执行。
可以通过发送 SIGSTOP 信号给进程来停止(T)进程。这个被暂停的进程可
以通过发送 SIGCONT 信号让进程继续运行。
Z:表示僵死状态(Zombie),该进程的执行已经结束,但其父进程尚未处理终止状态。
<:表示进程在前台执行。
N:表示进程在后台执行。
X:死亡状态(dead):这个状态只是一个返回状态,你不会在任务列表里看到这个状态。
除了上述符号外,ps
命令和top
命令还可以提供更多的信息,如PID(进程ID)、USER(进程所有者)、%CPU(CPU占用率)、%MEM(内存占用率)等。可以使用命令的不同选项或参数来定制所需的输出。
需要注意的是,不同版本的Linux发行版可能会有轻微的差异,因此具体符号和标识可能会有所不同。在查看进程状态时,建议参考相关文档或手册以获取准确的信息。
当一个进程执行完毕后,它的退出状态和其他信息需要被保留,以供其父进程查询。这是为了确保父进程可以获取有关子进程执行结果的信息。在Linux系统中,当一个进程结束但其父进程尚未通过wait()
或waitpid()
函数来获取这些信息时,被称为僵尸进程(Zombie)。
僵尸进程并不具有实际的执行代码,它们只是在进程表中保留了一些元数据,如进程ID(PID)、退出状态等。这些信息可以通过系统调用wait()
或waitpid()
来获取,以告知父进程子进程的结束状态。当父进程成功获取这些信息后,操作系统会清理掉僵尸进程的数据,并释放相关资源。
僵尸进程产生的主要原因是父进程没有及时处理子进程的退出状态。这可能是由于父进程繁忙或者因为设计上的问题导致的。僵尸进程本身并不会对系统造成直接影响,因为它们不占用计算机资源。然而,如果系统中存在大量僵尸进程,可能会消耗掉进程表中的可用项,并最终导致系统资源不足。
为了避免僵尸进程的产生,父进程应该正确处理子进程的退出状态。通常的做法是父进程在创建子进程后,通过wait()
或waitpid()
函数等待子进程的退出,并及时处理其状态。这样可以确保子进程的资源得到释放,避免产生大量僵尸进程。
另外,操作系统也提供了一些机制来自动处理僵尸进程,如SIGCHLD信号和守护进程。通过正确使用这些机制,可以有效地预防和清理僵尸进程。
如上便是本期的所有内容了,如果喜欢并觉得有帮助的话,希望可以博个点赞+收藏+关注❤️ ,学海无涯苦作舟,愿与君一起共勉成长