一、僵死进程的定义
一个进程执行结束也就是代码部分执行结束,但是进程的PCB没有被系统释放。
进程结束后,在PCB中还要保存进程退出码,以备其父进程获取其退出码。而僵死进程恰恰就是父进程没有获取退出码。也就是父进程未结束,子进程结束,并且父进程没有获取子进程的退出码,这样就形成了僵死进程。
二、在系统上体现僵死进程
我们在系统上创建这样一个文件,里面写这样的代码
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <assert.h>
int main()
{
pid_t pid = fork();
assert(pid != -1);
if(pid == 0)
{
printf("child starting\n");
sleep(10);
printf("child over\n");
}
else
{
printf("father starting\n");
sleep(20);
printf("father over\n");
}
exit(0);
}
从上面的代码我们看到父进程比子进程多执行10秒钟,在这个10秒的时间差内。父进程就是一个僵死进程。我们在系统上进行运行后的结果是:
运行之后我们可以看到第一次查看是当父子进程还在执行的时候,系统上是存在父进程和子进程的。在等待10秒之后,子进程执行结束后。再去查看时,这个时候我们发现系统上就出现了僵死进程。如上图中的红色部分,这个符号是僵死进程的标识。
三、如何处理僵死进程的产生
僵死进程的危害是比较大的。像我们刚写的代码,它的大小可能才1K到2K之间,但是一个僵死进程的大小就是1.7K。代码是在磁盘上的,但是僵死进程是在内存上的。如果大量的僵死进程堆积在内存上,系统内存将会很快的被耗尽,所以我们要避免僵死进程的产生,并且在它出现后能很快得处理僵死进程。
那么如何处理僵死进程呢?
很简单,我们只需要让父进程获取子进程的退出状态
方法如下:
pid_t wait(int *result);
这个函数返回处理的进程的PID。
result:获取进程的退出码
就如刚才的代码,我们如何处理刚才产生的僵死进程呢?我们只需要在代码上加上这样两句代码:
pid_t id=wait(NULL);
printf(“id = %d\n”,id);
代码如下:
int main()
{
pid_t pid = fork();
assert(pid != -1);
if(pid == 0)
{
printf("child starting: %d\n",getpid());
sleep(10);
printf("child over\n");
}
else
{
pid_t id=wait(NULL);
printf("id = %d\n",id);
printf("father starting\n");
sleep(20);
printf("father over\n");
}
exit(0);
}
再在系统上执行的时候我们就会发现等child over以后,系统中就只剩一个进程。调用wait后会有两个问题的存在
第一个是调用wait后,wait本身会阻塞,直到任意一个子进程退出,然后获取子进程的退出状态。
第二个假如有三个子进程,那么在使用wait时会调三次wait,而且三个进程的结束时间不一样,父进程就会去等待子进程的结束