目录
[2.1 进程状态](#2.1 进程状态)
[2.3.2PRI and NI](#2.3.2PRI and NI)
一、前言
在这一章,会为大家进行进程状态以及进程优先级的讲解
二、正文
2.1 进程状态
在一节中我们简单了描述了下进程的概念,进程粗略的来说就是由一个内核PCB数据结构加上我们自己的数据和代码。同时,我们也学会了如何通过指令来查看进程的相关属性。我们通过ps axj就可以查看所有进程的属性。
通过屏幕显示出来的数据,我们已经认识了PID,PPID,也就是我们讲的父进程和子进程对应的ID,但是我们还发现了一个属性"STAT",它的全程是station,即状态。所以每一个进程都是有其对应的状态,那么对于进程的状态我们应该如何去认识呢?
博主在网上找了些进程状态的图,小伙伴们可以先看看,有什么想法?
我们会发现,如果我们去网上去搜索关于进程状态的图片,其实都没有一个准确的说法,那么今天我们该如何去认识进程的状态呢?
首先,我们先来从操作系统学科的角度来谈谈进程的状态。无论网上的图片有多少状态,教材上有多少状态,笔者认为我们可以先将它们简单地归为三种状态:运行、阻塞和挂起。
1.运行状态------R
对于CPU来说,当它面临很多的进程的时候,它会创建一个运行队列,将要运行的进程一个个链入到队列当中,那么只要一个进程位于这个运行队列的时候,这个时候它的状态就属于运行态,即告诉CPU,它这个进程已经准备好了,可以随时被调度。因此,有些小伙伴会认为一个进程只有当它正在运行的时候才处于运行状态,这个想法其实是不太对的。
2.阻塞状态------S
在讲解阻塞状态之前,我们先要了解一个叫做时间片的概念,即CPU分配给每个进程的时间,可能是10ms,也可能是其他的时间,那么这个时间片有什么用呢?有了这个时间片,就能能够保证在一段时间内,所有的进程代码都会被执行,也叫做并发执行,而不会因为一直在等待某一个进程,让其他进程一直无法被执行。因此随着时间一到,进程就会被CPU放上去或者拿下来,那么大量的把进程从CPU放上去,拿下来的动作就叫做进程切换。
讲了这么多,那么到底什么是阻塞状态呢?我们都知道操作系统是一个可以帮助用户去管理软硬件的软件,那么除了能够帮助我们去管理不同的进程,它还要去管理各种的设备,外设,因为有些进程需要去访问这些硬件资源。相信,在座的小伙伴一定面临过这种情况,当我们在学习C语言的时候,有一个函数叫做scanf,这个函数能够帮助我们从键盘去读取数据,但是当我们运行程序的时候,我们一直不输入数据的话,终端是不是就一直进行等待,直至我们输入数据。也就是此时进程在等待键盘的这个外设的数据,那么如果我一直不输入数据,CPU就要一直等吗?结果是显然的,并不会,这也就是为什么我们前面要时间片这一个概念。
所以除了有运行队列之外,还有有一个等待队列的存在。就拿刚刚来说,键盘这个PCB中的等待队列就会把scanf这个进程链入,直至我把数据输入进去。 因此位于等待队列的进程我们就称它此时处于阻塞状态。
3.挂起状态------D
什么时候一个进程会处于挂起状态呢?挂起状态(Suspended State)在操作系统中是指进程被暂时移出内存,停止执行的状态。当系统的资源不足时,操作系统会将某些进程挂起,以释放内存供其他进程使用。挂起的进程会在条件允许时被重新激活,移回内存并继续执行。
但是以上三种状态是基于操作系统学科的角度上,实际上Linux的实际进程的状态并不是这样,而是在它们的基础上又进行了更多的演化,下面就让我们来认识下Linux下的各种进程状态
R/R+状态
当我们的程序一直在循环的时候,我们查询这个进程就会发现它处于运行状态------R,说明这个进程处于我们的运行队列中,那为什么它的后面还有一个+号,说明则是这个进程是在前台进行的。
cpp
1 #include <stdio.h>
2 #include <unistd.h>
3
4 int main()
5 {
6 while(1) {}
7 }
bash
//进程循环
./proc
如果没有加号,则说明当前进程是在后台进行的,我们只需在执行程序的时候加上&即可
bash
//在后台执行进程
./ proc &
S状态
当Linux中的某个进程处于阻塞状态时,进程就会处于S状态,也叫做浅度睡眠。
当我们执行下面这个代码时,虽然我们在不断的打印,但是由于CPU的速度是很快的,屏幕的显示跟不上,因此该进程就一直在等待屏幕的输出,就会处于阻塞状态
cpp
//循环打印
1 #include <stdio.h>
2 #include <unistd.h>
3
4 int main()
5 {
6 while(1)
7 {
8 printf("Hello Linux\n");
9 }
10 }
bash
//循环打印
./proc
bash
//后台循环打印
./proc &
D状态
D状态又称深度睡眠状态,那么它与我们之前讲解的挂起状态有什么关联呢?我们设想这样一个场景,当一个进程正在向磁盘写入数据时,由于写入的速度是远远慢于cpu的读取速度,因此该进程按理说应该处于我们刚刚所讲的阻塞状态即S状态。但是,恰好此时CPU管理的进程太多了,导致其不得不挪出一些进程,甚至是杀掉一些进程。但是如果我们将上面那个阻塞进程杀掉后,数据不就丢失了吗?这种情况明显不是我们用户希望看到的,因此Linux引入了一个新的状态------D状态,当一个进程处于该状态时,它就不能被CPU杀掉了,也就不会丢失数据了。
T/t状态
T/t状态是一种停止状态,通过kill指令中的选项-19,我们就可以让一个进程停止进入T状态或者当我们调试某个程序时,触发断点处,就会使该进程进入t状态。要注意的是对于S和D状态,它们对应进程一定是在等待资源,但是T/t状态的进程可能是等待资源也可能是单纯停止不让该进程继续运行。
cpp
//停止指定进程
kill -19 +指定进程的PID
X状态
X状态------终止态,处于该状态的进程,说明该进程已终止。但是由于该状态并不方便观察,因此只需了解该状态即可。
Z状态
实际上,虽然在上面我们讲了X状态,但是当一个进程在死的时候其并不会立马进入X状态,而是会进入到Z状态------僵尸状态,而进入僵尸状态的进程我们也叫做僵尸进程。
在详细讲解Z状态之前,我们需要再新认识一个系统函数接口------exit(),通过该接口我们结束一个进程,下面是其说明手册,通过包含其头文件,我们便可以调用该接口了
在执行下述代码的时候,我们发现虽然子进程已经调用exit接口结束后,但是该进程尽没有进入到终止态,而是进入了僵尸态,这是由于子进程虽然退出了,父进程并没有主动回收子进程信息,因此子进程会一直让自己处于Z状态,于是子进程的相关资源尤其是task_struct结构体不能释放,也就导致了一个很严重的问题------内存泄漏。因此当子进程结束后,一定要对其进行回收,避免内存泄漏,至于如何回收会在后面讲到。
cpp
1 #include <stdio.h>
2 #include <unistd.h>
3 #include <sys/types.h>
4 #include <stdlib.h>
5
6 int main()
7 {
8
9 int ret= fork();
10 if(ret==0)
11 {
12 printf("我是子进程,PPID:%d,PID:%d\n",getppid(),getpid());
13 exit(0);
14 }
15 if(ret>0)
16 {
17 while(1)
18 {
19 printf("我是父进程,PPID:%d,PID:%d\n",getppid(),getpid());
20 }
21 }
22 return 0;
23 }
2.2孤儿进程
关于Linux值得关注的几种进程状态我们已经讲述完毕了,接下来我们来关注一类比较特殊的进程------孤儿进程。
那么什么是孤儿进程,或者它是如何产生的?正常情况,对于父进程和子进程来说,子进程应该是要先于父进程退出的,因为它需要父进程来完成对它的资源清理。但是,如果有一天父进程先于子进程退出,那么子进程就失去了父亲,自然就成了孤儿,于是也就有了"孤儿进程"一说。但是这样它的资源就无从释放,于是我们就想着能不能给它再找个父亲,于是我们选取了一个最优的父亲,也就是PID为1号的init进程来领养它,由该进程对其进行回收。
2.3进程优先级
到目前为止,对于进程的属性,我们已经认识了PID,PPID,STAT,但是当我们进行进行进程属性的显示时,我们还发现了PRI和NI这两个属性,这两个属性就与我们这小节所要讲的进程优先级相关。
2.3.1基本概念
cpu资源分配的先后顺序,就是指进程的优先权(priority)。
优先权高的进程有优先执行权利。配置进程优先权对多任务环境的linux很有用,可以改善系统性能。
还可以把进程运行到指定的CPU上,这样一来,把不重要的进程安排到某个CPU,可以大大改善系统整 体性能。
2.3.2PRI and NI
PRI也还是比较好理解的,即进程的优先级,或者通俗点说就是程序被CPU执行的先后顺序,此值越小 进程的优先级别越高
那NI呢?就是我们所要说的nice值了,其表示进程可被执行的优先级的修正数值
PRI值越小越快被执行,那么加入nice值后,将会使得PRI变为:PRI(new)=PRI(old)+nice
这样,当nice值为负值的时候,那么该程序将会优先级值将变小,即其优先级会变高,则其越快被执行
所以,调整进程优先级,在Linux下,就是调整进程nice值 nice其取值范围是-20至19,一共40个级别
查看当前目录下进程的优先级
ps -l
查看该用户创建的所有进程的优先级
ps -al
在上面的例子中我们可以看到,进程的PRI都是80,NI都为0,也就是说他们的优先级都为80。那么优先级可不可以被调整呢?答案当然是可以的,通过调整某一个进程的优先级,就可以让我们的进程可以几乎一直处于被调度的状态,但是这样明显是操作系统不愿意看到的,因为在Linux中,不想过多的让用户参与优先级的调整,所以一般情况下我们是不需要对进程的优先级进行调整的。
但是我们可以了解一些对应调整优先级的指令。
top命令
●top
●进⼊top后按"r"‒>输⼊进程PID‒>输⼊nice值
●按"q"退出,及修改完成
nice命令nice命令用于在启动一个新进程时指定其优先级,即进程的"nice值"。Nice值是一个整数,范围从-20(最高优先级)到19(最低优先级),默认情况下,大多数进程的nice值为0。
语法:nice [选项] [nice值] 命令 [参数]
renice命令renice命令用于修改已经运行的进程的优先级。
语法:renice [选项] 优先级 进程ID
三、结语
到此为止,关于进程状态和进程优先级的讲解就暂时告一段落啦!