Linux操作系统和进程基本概念(下)

简介: Linux操作系统和进程基本概念

🎵三、进程状态


为了弄明白正在运行的进程是什么意思,我们需要知道进程的不同状态,而一个进程可以有多少种状态呢?我们可以查看一下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状态:运行状态。


1669271858654.jpg


这里需要阐明几点:


1、一个CPU在任何时刻只能运行一个进程,一个CPU一个运行队列。


2、让进程进入运行队列的本质就是将该进程的task_struct 结构体对象放入运行队列中。


3、进程PCB只要在runqueue,就是R状态,并不一定是这个进程在运行才是运行状态。

1669272151359.jpg


🎈Ⅱ S状态


👓①阻塞状态


S状态意味着进程在等待时间完成。比较常见的S状态就是阻塞状态。


那么什么是阻塞状态呢?很简单,我们的进程不会只占用CPU资源,它可能随时随地地占用外设,比如我们要打印一些字符到显示器,就是占用外设。那么我们都知道CPU的运行速度是远大于外设的,如果我们的进程需要访问外设而外设较慢,那么此时CPU会停下来等我们访问外设结束吗?这样显然太过于浪费CPU资源了,此时,为了追求效率,操作系统会把该进程连接到外设的等待队列,然后CPU调用执行别的进程。那么此时进程就不是R状态,而是阻塞状态,也称为S状态。


当然,当进程访问外设结束的时候,操作系统会得知然后把进程状态由阻塞(S)改为R状态,放在CPU队列。


1669272165631.jpg


可以看到这里我们需要打印到显示器就是进程在访问使用外设,这里是死循环不停调用外设显示器,当然中间也会调用CPU,但是访问外设可能占用99%的时间,剩下的1%时间调用CPU,所以我们查询显示一般都是S状态。


👓②挂起状态


什么是挂起状态呢?我们都知道内存空间是有限的,如果我们的内存空间充满了大量阻塞的进程导致内存空间不够用这时候怎么办呢?不够的空间我们要找哪里去借空间,去哪里呢?磁盘就是一个不错的地方,它的空间很大!没错,我们的操作系统也是这么处理的,它会把阻塞进程的代码和数据暂时保存到磁盘上为其他进程腾出空间。这种要等待很长时间,被暂时保存到磁盘的进程状态称为挂起。


1669272193002.jpg


🎈 ⅢT状态


T停止状态:可以通过发送SIGSTOP信号给进程来停止(T)进程,这个被暂停的进程可以通过SIGCONT信号让进程继续运行。


首先我们要搞清楚这两个信号是什么鬼?不知道老铁是否还有印象我上文提到的两个没有细说的kill指令呢?

1669272206916.jpg

没错,就是kill -19 和kill -18指令。


kill -19 指令停止进程。


1669272216017.jpg


kill -18 继续进程


1669272230047.jpg


差点忘记,这里还有一个点没说。不知道老铁注意到没有我们之前的进程状态是R+状态,现在是R状态,为什么少了+号,以及+号有什么用处呢?


🖊状态后带有+号的是前台,我们对它的输入无效,只能通过CTRL+c或者kill -9 终止进程。


🖊状态后没有+号的是后台,我们输入的指令有效,但是仍在运行程序,CTRL+c也不能终止,只能通过kill -9 终止进程。


演示:


1669272243521.jpg



通过演示相信老铁感受到了后台和前台的不同。


🎈ⅣD状态


D磁盘休眠状态有时候也叫不可中断睡眠状态,在这个状态的进程通常会等待IO的结束。


D状态叫做深度休眠,那它和S状态有什么区别吗?S状态是浅度睡眠,能够被终止,而D状态是不可被终止的。


那么D状态的应用情景是什么呢?我们有没有这样的一种情景:当操作系统严重内存不足,就连挂起也解决不了的时候,操作系统会采取终止进程的操作,否则操作系统会挂掉。而我们不想让操作系统挂掉某些重要的进程,就把这些进程标识为D状态,避免被终止。在该状态的进程无法被OS杀掉,只能通过断电,或者进程自己醒来来解决。


这种情形只有高IO的情形下才会出现。


🎈Ⅴ t状态


t (tracing sleep)状态 也是一种暂停状态。调试等待运行,当我们调试运行到断点处时我们发现gdb调试的状态为t状态。


插播一个小tip:


1669272284401.jpg


makefile的小技巧


1669272293715.jpg


🎈Ⅵ Z状态


Z(zombie)-僵尸进程。


🖊僵死状态是一种比较特殊的状态。当进程退出但是父进程或操作系统没有读取到子进程退出的返回代码时就会产生僵尸进程。


🖊僵尸进程会以终止状态保持在进程表中,并且会一直在等待父进程或操作系统读取退出状态代码。


🖊所以,只要子进程退出,父进程或操作系统还在运行,但父进程或操作系统没有读取子进程状态,子进程进入Z状态。


我们这里写一个脚本指令循环监控显示。


1669272309803.jpg


我们写一个函数,让父进程循环,子进程运行后就停止。


1669272318625.jpg


运行查询进程状态:


1669272327249.jpg


因为我们的代码是子进程只打印一次就停止,父进程是不断循环打印,所以父进程未结束而子进程结束时父进程处于S+状态,而子进程处于Z+状态。


当然,我们不只这种方式使子进程先停止,如果我们使用kill指令终止子进程,子进程也会处于僵尸状态等待父进程运行结束。


1669272336374.jpg


僵尸进程的危害:


🖊进程的退出状态只要不被父进程或操作系统读取就会一直维持僵尸状态下去,内存也一直要分配空间给它。那么要是一个父进程创建很多子进程,而子进程早早结束父进程不回收就会造成内存资源的浪费,可能造成内存泄漏。


至此,我们的进程有三种状态是无法被杀死的状态:D X Z


🎈Ⅶ 孤儿进程


之前是子进程先于父进程结束而处于Z状态,那么有没有可能父进程先结束呢?有的,这种父进程提前结束的子进程被称为孤儿进程。


1669272351663.jpg


注意啊,这里并不是在一个窗口执行指令的,我是同一个用户Gyh开了三个窗口,一个窗口执行运行进程,一个窗口查询进程状态,一个窗口执行杀死父进程指令。


1669272404775.jpg


这里有一个疑问,我们在杀掉父进程之后,为什么不显示父进程为僵尸状态(Z)呢?原因很简单,因为父进程也有它的父进程,在它的父进程显示为Z状态,这里我们更加关注子进程。我们发现子进程有父进程,它的父进程pid为1,那么这个父进程是谁呢?


1669272374069.jpg


🖊我们要明白,父进程先结束这种现象一定是存在的,而子进程也会被操作系统(1号进程)领养,如果不领养,那么子进程退出的时候,对应的僵尸进程,便无法回收了。


我们这里还发现孤儿进程的状态由S+状态变为S状态,由前台切换为后台,所以要想杀死,得用kill -9 指令杀死。


🎵四、进程优先级


🎈Ⅰ基本概念


进程在都能被执行的情况下,我们先做哪个进程和它的优先级有一定关系,优先级高的进程先执行。这个道理很容易明白,因为我们的CPU资源总是有限的,而进程的数量可能很多,我们优先执行比较重要的进程。配置进程优先权对多任务环境的Linux很有用,可以改善系统性能。而Linux优先级的本质也就是PCB里面的整数数字。


🎈Ⅱ查看进程优先级


我们在Linux下使用ps -la 指令会输出以下内容:


1669272427521.jpg


除了我们已经介绍过的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值


演示:


1669272441986.jpg



🎵五、其他概念


🖊竞争性:系统进程数目众多,而CPU资源只有少量,比较老的甚至只有一个,所以进程之间是有竞争属性的。为了高效完成任务,更合理竞争相关资源,便有了优先级。


🖊独立性:多进程运行,需要独享各种资源,多进程运行期间互不干扰


🖊并行:多个进程在多个CPU下分别,同时进行运行,这称之为并行。


🖊并发:多个进程在一个CPU下采用进程切换的方式,在一段时间之内,让多个进程都得以推进,称之为并发。


这里其他都好理解,需要着重强调的是并发的概念。


这里一个CPU时,一个进程并不是直到运行完才被拿下来,单CPU采用时间轮转片的策略,不管进程执行完花多长时间,每个进程比如说只给10ms的时间在CPU上运行,时间到了不管有没有运行结束都必须从CPU上剥离下来,然后放在运行队列,多个进程不断切换,实现单CPU在一个时间段内多个进程同时进行,我们称为并发。


这是具体并发是如何单核实现多进程的。当然这其中还有很多细节需要阐述。


①如果进程没有运行结束就被剥离下来,那么再次被放到CPU上是重新开始运行,还是继续执行之前的进度,如果执行之前的进度,那么产生的临时数据文件被寄存在哪里呢?


②单CPU在任一时间段进行多进程,那么任一时刻呢?


🖊先来回答第一个问题:首先我们要明白单个CPU只有一套寄存器,而且它的空间很小,负责当前正在运行的进程。它当然是继续之前的进度,那么它的临时数据不是由寄存器保管,记住,寄存器只保存当前进程的数据,不会去保存之前进程的数据,之前进程的数据专门保存在内存的某一特定区域。寄存器被所有的进程共享,但是寄存器内的数据,是每个进程私有的,仔细品品。


🖊第二个问题很简单,任一时刻单个CPU只能进行一个进程。

相关文章
|
4月前
|
NoSQL Linux 编译器
GDB符号表概念和在Linux下获取符号表的方法
通过掌握这些关于GDB符号表的知识,你可以更好地管理和理解你的程序,希望这些知识可以帮助你更有效地进行调试工作。
186 16
|
4月前
|
Web App开发 Linux 程序员
获取和理解Linux进程以及其PID的基础知识。
总的来说,理解Linux进程及其PID需要我们明白,进程就如同汽车,负责执行任务,而PID则是独特的车牌号,为我们提供了管理的便利。知道这个,我们就可以更好地理解和操作Linux系统,甚至通过对进程的有效管理,让系统运行得更加顺畅。
115 16
|
4月前
|
Unix Linux
对于Linux的进程概念以及进程状态的理解和解析
现在,我们已经了解了Linux进程的基础知识和进程状态的理解了。这就像我们理解了城市中行人的行走和行为模式!希望这个形象的例子能帮助我们更好地理解这个重要的概念,并在实际应用中发挥作用。
93 20
|
3月前
|
监控 Shell Linux
Linux进程控制(详细讲解)
进程等待是系统通过调用特定的接口(如waitwaitpid)来实现的。来进行对子进程状态检测与回收的功能。
74 0
|
3月前
|
存储 负载均衡 算法
Linux2.6内核进程调度队列
本篇文章是Linux进程系列中的最后一篇文章,本来是想放在上一篇文章的结尾的,但是想了想还是单独写一篇文章吧,虽然说这部分内容是比较难的,所有一般来说是简单的提及带过的,但是为了让大家对进程有更深的理解与认识,还是看了一些别人的文章,然后学习了学习,然后对此做了总结,尽可能详细的介绍明白。最后推荐一篇文章Linux的进程优先级 NI 和 PR - 简书。
102 0
|
3月前
|
存储 Linux Shell
Linux进程概念-详细版(二)
在Linux进程概念-详细版(一)中我们解释了什么是进程,以及进程的各种状态,已经对进程有了一定的认识,那么这篇文章将会继续补全上篇文章剩余没有说到的,进程优先级,环境变量,程序地址空间,进程地址空间,以及调度队列。
65 0
|
3月前
|
Linux 调度 C语言
Linux进程概念-详细版(一)
子进程与父进程代码共享,其子进程直接用父进程的代码,其自己本身无代码,所以子进程无法改动代码,平时所说的修改是修改的数据。为什么要创建子进程:为了让其父子进程执行不同的代码块。子进程的数据相对于父进程是会进行写时拷贝(COW)。
67 0
|
6月前
|
Linux 数据库 Perl
【YashanDB 知识库】如何避免 yasdb 进程被 Linux OOM Killer 杀掉
本文来自YashanDB官网,探讨Linux系统中OOM Killer对数据库服务器的影响及解决方法。当内存接近耗尽时,OOM Killer会杀死占用最多内存的进程,这可能导致数据库主进程被误杀。为避免此问题,可采取两种方法:一是在OS层面关闭OOM Killer,通过修改`/etc/sysctl.conf`文件并重启生效;二是豁免数据库进程,由数据库实例用户借助`sudo`权限调整`oom_score_adj`值。这些措施有助于保护数据库进程免受系统内存管理机制的影响。
|
6月前
|
Linux Shell
Linux 进程前台后台切换与作业控制
进程前台/后台切换及作业控制简介: 在 Shell 中,启动的程序默认为前台进程,会占用终端直到执行完毕。例如,执行 `./shella.sh` 时,终端会被占用。为避免不便,可将命令放到后台运行,如 `./shella.sh &`,此时终端命令行立即返回,可继续输入其他命令。 常用作业控制命令: - `fg %1`:将后台作业切换到前台。 - `Ctrl + Z`:暂停前台作业并放到后台。 - `bg %1`:让暂停的后台作业继续执行。 - `kill %1`:终止后台作业。 优先级调整:
318 5
|
运维 关系型数据库 MySQL
掌握taskset:优化你的Linux进程,提升系统性能
在多核处理器成为现代计算标准的今天,运维人员和性能调优人员面临着如何有效利用这些处理能力的挑战。优化进程运行的位置不仅可以提高性能,还能更好地管理和分配系统资源。 其中,taskset命令是一个强大的工具,它允许管理员将进程绑定到特定的CPU核心,减少上下文切换的开销,从而提升整体效率。
掌握taskset:优化你的Linux进程,提升系统性能

热门文章

最新文章