Linux–僵死进程(僵尸进程)

1.僵死进程产生的原因或者条件:

当子进程先于父进程结束,父进程没有获取子进程的退出码,此时子进程变成僵死进程;(即就是子进程结束了,但父进程还没有结束的时候才会出现僵死进程)

(代码中 ,子先于父)(& 后台运行)

 

当一个进程结束的时候,只有进程的退出码被父进程获取后,父进程才能退出,进程的PCB才会被释放,否则父进程会一直等待退出码,在得到退出码之前,这个进程就是一个僵死进程。

2.演示僵死进程:

vim fork.c

#include<stdlib.h>
#include<wait.h>int main()
{char *s=NULL;int n=0;//控制父子进程的次数pid_t id=fork();assert(id!=-1);if(id==0)//子进程{   s="child";n=7;}   else{   s="parent";n=3;int val=0;if(WIFEXITED(val))//如果正常退出{printf("val=%dn",WEXITSTATUS(val));}wait(&val);printf("val=%dn",val);}   //父子进程一起运行int i=0;for(;i<n;i++){   printf("s=%s ,pid=%d ,ppid=%d n的地址=%p n=%dn",s,getpid(),getppid(),&n,n);sleep(1);}   exit(0);
}

 3.僵死进程带来的影响

(1)进程的退出状态必须被维持下去,因为他要告诉关心它的进程(父进程),你交给我的任务,我办的怎 么样了。可父进程如果一直不读取,那子进程就一直处于Z状态。

(2)维护退出状态本身就是要用数据维护,也属于进程基本信息,所以保存在task_struct(PCB)中,换句话 说,Z状态一直不退出,PCB一直要维护

(3)那一个父进程创建了很多子进程,就是不回收,是不是就会造成内存资源的浪费?

答:是的。因为数据结构 对象本身就要占用内存,想想C中定义一个结构体变量(对象),是要在内存的某个位置进行开辟空 间

(4)会造成内存泄漏

4.如何去处理僵死进程?

(1)父进程先结束

比如子进程的n改为7,父进程的n改为3;

(子进程会变成孤儿进程,孤儿进程会被收养,"父进程"就会获取退出码;(调用wait);

(红帽子是init(pid==1)收养孤儿进程,Unbuntu是操作系统指定了一个进程));

(2)父进程调用wait()方法获取子进程的退出码;

父进程执行7,子进程改为3;就能观察到僵死进程(放到后台运行&,ps);

#include<stdlib.h>
#include<wait.h>int main()
{char *s=NULL;int n=0;//控制父子进程的次数pid_t id=fork();assert(id!=-1);if(id==0)//子进程{   s="child";n=7;}   else{   s="parent";n=3;int val=0;if(WIFEXITED(val))//如果正常退出{printf("val=%dn",WEXITSTATUS(val));}wait(&val);printf("val=%dn",val);}   //父子进程一起运行int i=0;for(;i<n;i++){   printf("s=%s ,pid=%d ,ppid=%d n的地址=%p n=%dn",s,getpid(),getppid(),&n,n);sleep(1);}   exit(0);
}

处理僵死进程的方法一:

父进程执行3次,子进程改为7次;去观察僵死进程没有了;收养孤儿进程 的,一定会调用wait(就是获取退出码的);获取了退出码,操作系统就会把

pcb删掉,就不会出现僵死进程;

处理僵死进程的方法二:

父进程调用wait方法;

1命令, 2 系统调用 3 库函数

man 2 wait

else
{s="parent";n=7;int val=0;wait(&val);printf("val=%dn",val); 
}

其实两种处理僵死进程的方法的本质是一样的,都调用了wait方法,获取退出码;

但是两种方法都有区别:就是父进程调用wait会阻塞,等子进程执行完之后,父进程才会去执行

  1 #include <stdio.h>2 #include<assert.h>3 #include<unistd.h>4 #include<string.h>5 #include<stdlib.h>6 #include<wait.h>7 8 int main()9 {10     char *s=NULL;11     int n=0;//控制父子进程的次数12 13     pid_t id=fork();14     assert(id!=-1);15 16     if(id==0)//子进程17     {18         s="child";19         n=3;20     }21     else22     {23         s="parent";24         n=7;25         int val=0;26         wait(&val);27         printf("val=%dn",val);28     }//获取退出码29     //父子进程一起运行30     int i=0;31     for(;i<n;i++)32     {33         printf("s=%s ,pid=%d ,ppid=%d n的地址=%p n=%dn",s,getpid(),getppid    (),&n,n);34         sleep(1);35     }36     exit(0);37 }

5.子进程的退出码

改写码:exit(3)

执行显示val=768;

 man 2 wait

 

WIFEXITED:正常退出的状态;

WEXITSTATUS:获取退出码;

 

Published by

风君子

独自遨游何稽首 揭天掀地慰生平