🎵三、进程状态
为了弄明白正在运行的进程是什么意思,我们需要知道进程的不同状态,而一个进程可以有多少种状态呢?我们可以查看一下Linux内核代码的定义:
/* * The task state array is a strange "bitmap" of * reasons to sleep. Thus "running" is zero, and * you can test for combinations of others with * simple bit tests. */ 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 */ };
我们可以看到后面的注释,没错,其实进程状态没老铁想的那么复杂,在PCB内部就是简单的整数表示的。
查看进程状态的指令是 ps aux / ps ajx / ps axj 这些都可以。
🎈ⅠR状态
要了解什么是R状态,我们首先要明白进程具体在CPU如何执行的。
在操作系统CPU有它的运行队列,在这个队列中都是等待被CPU执行代码的进程,而正在被CPU处理或处于运行队列的进程所处的状态都称为R状态:运行状态。
这里需要阐明几点:
1、一个CPU在任何时刻只能运行一个进程,一个CPU一个运行队列。
2、让进程进入运行队列的本质就是将该进程的task_struct 结构体对象放入运行队列中。
3、进程PCB只要在runqueue,就是R状态,并不一定是这个进程在运行才是运行状态。
🎈Ⅱ S状态
👓①阻塞状态
S状态意味着进程在等待时间完成。比较常见的S状态就是阻塞状态。
那么什么是阻塞状态呢?很简单,我们的进程不会只占用CPU资源,它可能随时随地地占用外设,比如我们要打印一些字符到显示器,就是占用外设。那么我们都知道CPU的运行速度是远大于外设的,如果我们的进程需要访问外设而外设较慢,那么此时CPU会停下来等我们访问外设结束吗?这样显然太过于浪费CPU资源了,此时,为了追求效率,操作系统会把该进程连接到外设的等待队列,然后CPU调用执行别的进程。那么此时进程就不是R状态,而是阻塞状态,也称为S状态。
当然,当进程访问外设结束的时候,操作系统会得知然后把进程状态由阻塞(S)改为R状态,放在CPU队列。
可以看到这里我们需要打印到显示器就是进程在访问使用外设,这里是死循环不停调用外设显示器,当然中间也会调用CPU,但是访问外设可能占用99%的时间,剩下的1%时间调用CPU,所以我们查询显示一般都是S状态。
👓②挂起状态
什么是挂起状态呢?我们都知道内存空间是有限的,如果我们的内存空间充满了大量阻塞的进程导致内存空间不够用这时候怎么办呢?不够的空间我们要找哪里去借空间,去哪里呢?磁盘就是一个不错的地方,它的空间很大!没错,我们的操作系统也是这么处理的,它会把阻塞进程的代码和数据暂时保存到磁盘上为其他进程腾出空间。这种要等待很长时间,被暂时保存到磁盘的进程状态称为挂起。
🎈 ⅢT状态
T停止状态:可以通过发送SIGSTOP信号给进程来停止(T)进程,这个被暂停的进程可以通过SIGCONT信号让进程继续运行。
首先我们要搞清楚这两个信号是什么鬼?不知道老铁是否还有印象我上文提到的两个没有细说的kill指令呢?
没错,就是kill -19 和kill -18指令。
kill -19 指令停止进程。
kill -18 继续进程
差点忘记,这里还有一个点没说。不知道老铁注意到没有我们之前的进程状态是R+状态,现在是R状态,为什么少了+号,以及+号有什么用处呢?
🖊状态后带有+号的是前台,我们对它的输入无效,只能通过CTRL+c或者kill -9 终止进程。
🖊状态后没有+号的是后台,我们输入的指令有效,但是仍在运行程序,CTRL+c也不能终止,只能通过kill -9 终止进程。
演示:
通过演示相信老铁感受到了后台和前台的不同。
🎈ⅣD状态
D磁盘休眠状态有时候也叫不可中断睡眠状态,在这个状态的进程通常会等待IO的结束。
D状态叫做深度休眠,那它和S状态有什么区别吗?S状态是浅度睡眠,能够被终止,而D状态是不可被终止的。
那么D状态的应用情景是什么呢?我们有没有这样的一种情景:当操作系统严重内存不足,就连挂起也解决不了的时候,操作系统会采取终止进程的操作,否则操作系统会挂掉。而我们不想让操作系统挂掉某些重要的进程,就把这些进程标识为D状态,避免被终止。在该状态的进程无法被OS杀掉,只能通过断电,或者进程自己醒来来解决。
这种情形只有高IO的情形下才会出现。
🎈Ⅴ t状态
t (tracing sleep)状态 也是一种暂停状态。调试等待运行,当我们调试运行到断点处时我们发现gdb调试的状态为t状态。
插播一个小tip:
makefile的小技巧
🎈Ⅵ Z状态
Z(zombie)-僵尸进程。
🖊僵死状态是一种比较特殊的状态。当进程退出但是父进程或操作系统没有读取到子进程退出的返回代码时就会产生僵尸进程。
🖊僵尸进程会以终止状态保持在进程表中,并且会一直在等待父进程或操作系统读取退出状态代码。
🖊所以,只要子进程退出,父进程或操作系统还在运行,但父进程或操作系统没有读取子进程状态,子进程进入Z状态。
我们这里写一个脚本指令循环监控显示。
我们写一个函数,让父进程循环,子进程运行后就停止。
运行查询进程状态:
因为我们的代码是子进程只打印一次就停止,父进程是不断循环打印,所以父进程未结束而子进程结束时父进程处于S+状态,而子进程处于Z+状态。
当然,我们不只这种方式使子进程先停止,如果我们使用kill指令终止子进程,子进程也会处于僵尸状态等待父进程运行结束。
僵尸进程的危害:
🖊进程的退出状态只要不被父进程或操作系统读取就会一直维持僵尸状态下去,内存也一直要分配空间给它。那么要是一个父进程创建很多子进程,而子进程早早结束父进程不回收就会造成内存资源的浪费,可能造成内存泄漏。
至此,我们的进程有三种状态是无法被杀死的状态:D X Z
🎈Ⅶ 孤儿进程
之前是子进程先于父进程结束而处于Z状态,那么有没有可能父进程先结束呢?有的,这种父进程提前结束的子进程被称为孤儿进程。
注意啊,这里并不是在一个窗口执行指令的,我是同一个用户Gyh开了三个窗口,一个窗口执行运行进程,一个窗口查询进程状态,一个窗口执行杀死父进程指令。
这里有一个疑问,我们在杀掉父进程之后,为什么不显示父进程为僵尸状态(Z)呢?原因很简单,因为父进程也有它的父进程,在它的父进程显示为Z状态,这里我们更加关注子进程。我们发现子进程有父进程,它的父进程pid为1,那么这个父进程是谁呢?
🖊我们要明白,父进程先结束这种现象一定是存在的,而子进程也会被操作系统(1号进程)领养,如果不领养,那么子进程退出的时候,对应的僵尸进程,便无法回收了。
我们这里还发现孤儿进程的状态由S+状态变为S状态,由前台切换为后台,所以要想杀死,得用kill -9 指令杀死。
🎵四、进程优先级
🎈Ⅰ基本概念
进程在都能被执行的情况下,我们先做哪个进程和它的优先级有一定关系,优先级高的进程先执行。这个道理很容易明白,因为我们的CPU资源总是有限的,而进程的数量可能很多,我们优先执行比较重要的进程。配置进程优先权对多任务环境的Linux很有用,可以改善系统性能。而Linux优先级的本质也就是PCB里面的整数数字。
🎈Ⅱ查看进程优先级
我们在Linux下使用ps -la 指令会输出以下内容:
除了我们已经介绍过的pid和ppid,我们Linux进程优先级主要和PRI(Priority)和NI(Nice)有关.那么它们又代表什么意思呢?
🖊PRI即进程的优先级,它的值越小,进程的优先级别越高。
🖊NI 就是我们要说的nice值,它表示进程可被执行的优先级的修正数值。
简单来说,Linux下有一套优先级计算公式:
最终优先级= 固定优先级数80 + nice值
也就是说Linux支持进程运行中,进行优先级调整的,调整的策略就是更改nice值。那么nice值是否可以任意修改呢?当然不是,它的范围是[-20,19]的整数,如果你输入大于19或者小于-20那么按照19和-20处理,也就是说我们最终优先级的范围有40个值,区间是[60,99]。
说了这么多,我们怎么更改进程优先级呢?
用top指令更改nice值。
top --> 按"r"--> 输入进程PID-->输入nice值
演示:
🎵五、其他概念
🖊竞争性:系统进程数目众多,而CPU资源只有少量,比较老的甚至只有一个,所以进程之间是有竞争属性的。为了高效完成任务,更合理竞争相关资源,便有了优先级。
🖊独立性:多进程运行,需要独享各种资源,多进程运行期间互不干扰
🖊并行:多个进程在多个CPU下分别,同时进行运行,这称之为并行。
🖊并发:多个进程在一个CPU下采用进程切换的方式,在一段时间之内,让多个进程都得以推进,称之为并发。
这里其他都好理解,需要着重强调的是并发的概念。
这里一个CPU时,一个进程并不是直到运行完才被拿下来,单CPU采用时间轮转片的策略,不管进程执行完花多长时间,每个进程比如说只给10ms的时间在CPU上运行,时间到了不管有没有运行结束都必须从CPU上剥离下来,然后放在运行队列,多个进程不断切换,实现单CPU在一个时间段内多个进程同时进行,我们称为并发。
这是具体并发是如何单核实现多进程的。当然这其中还有很多细节需要阐述。
①如果进程没有运行结束就被剥离下来,那么再次被放到CPU上是重新开始运行,还是继续执行之前的进度,如果执行之前的进度,那么产生的临时数据文件被寄存在哪里呢?
②单CPU在任一时间段进行多进程,那么任一时刻呢?
🖊先来回答第一个问题:首先我们要明白单个CPU只有一套寄存器,而且它的空间很小,负责当前正在运行的进程。它当然是继续之前的进度,那么它的临时数据不是由寄存器保管,记住,寄存器只保存当前进程的数据,不会去保存之前进程的数据,之前进程的数据专门保存在内存的某一特定区域。寄存器被所有的进程共享,但是寄存器内的数据,是每个进程私有的,仔细品品。
🖊第二个问题很简单,任一时刻单个CPU只能进行一个进程。