Bootstrap

Linux系列-僵尸状态

🌈个人主页:羽晨同学 

💫个人格言:“成为自己未来的主人~”  

进程退出

进程退出之后,代码就不会执行了,而是由PCB维护起来,我们可以通过PCB来查看退出信息。 

进程退出时首先可以立即释放的就是进程对应的程序信息数据。

进程退出,要有退出信息,也就是进程的退出码,保存在自己的task_struct内部。task_struct这个结构体里面包含的是成员属性,要有退出信息(int,code,其它)。

这个可以管理进程的结构(task_struct)必须被OS维护起来,方便用户未来进行获取进程退出的信息。就像下面的这个示例一样。

getpid() --> return curr->pid;

emmm,其实这个进程退出像什么呢?就像是自己从学校毕业,虽然自己人不在了,但是个人的信息还是在学校里面的。

这个个人信息就是进程的退出信息,在PCB里面。

所以,这个时候我们想一下进程的创建的过程,是先有代码和数据,还是先有对应的内核数据结构。

很明显的,是不是得先有对应的内核数据结构,当管理信息建立好的时候,才能放进去对应的代码和数据,就像是,你去上学,学校里面肯定是先有你的信息的,然后你的人再去那里。

对吧,那么在没有传入代码和数据的时候,进程是已经创建好的,但是啊,它不可以调度。

是不是很明显,其实这个时候你也就没什么可以调度的。

这个时候如果我们要给这个状态一个概念,就叫做僵尸状态

操作系统释放进程,先释放代码和数据,把task_struct维护起来,这个时候就叫做僵尸状态。

所以说,task_struct是最早产生,但是是最后删除的。

那么,如果我想要看到僵尸状态,我应该怎么设计测试用例呢?

我们创建出子进程,这个时候父子同时存在,然后我们让子进程退出,父进程还存活,但是父进程什么都不做(尽管我们也不知道它可以做什么)。

这个时候的子进程就会出现短暂的僵尸状态。

17228 17229 17228 15449 pts/2    17228 Z+    1000   0:00 [myprocess] <defunct>

 这个时候,这个进程就从S变成了Z,就是变成了僵尸状态。

所以,Z是进程的僵尸状态,这个状态是为了维护自己的task_struct。方便未来父进程读取子进程的退出状态。

这个时候我们使用kill命令来删除子进程,删除了子进程之后父进程是不会对子进程进行回收的。

所以这个时候子进程是无效的,失效的。

接下来,我们举一个删除前后子进程状态的对比。

15501 20317 20317 15449 pts/2    20317 S+    1000   0:00 ./myprocess

这个是删除前的。

20317 20318 20317 15449 pts/2    20317 Z+    1000   0:00 [myprocess] <defunct>

这个是删除之后的。

在这个时候,如果没有人处理这个子进程,那么这个子进程就会一直处于僵尸状态,这个子进程的task_struct就会一直存在,从而一直消耗内存,这就造成了内存泄漏

一般父进程是需要读取子进程信息的,一旦读取完了之后,子进程才会自动退出。

而一旦回收之后,这个进程也就没了。

我们是无法杀掉一个在概念上已经死掉的进程的,所以这个时候必须要父进程来进行回收。

malloc开辟的是属于数据的,当关闭这个进程的时候,数据会直接进行删除。若是有一个不退出的进程,常驻内存的进程,若是这个时候不断的malloc,那么会导致资源越来越少。

所以,其实对于语言层面的内存泄漏问题,若是对于那种会自动退出的程序而言,问题是不算太大的,若是发生在常驻内存的进程当中,这个时候问题会比较大。

而我们写的大部分软件都是常驻内存的。

好了,本次的文章就到这里了,我们下次再见。 

悦读

道可道,非常道;名可名,非常名。 无名,天地之始,有名,万物之母。 故常无欲,以观其妙,常有欲,以观其徼。 此两者,同出而异名,同谓之玄,玄之又玄,众妙之门。

;