目录
1.什么是僵死进程?
2.代码演示僵死进程
3.解决办法
1.什么是僵死进程?
- 僵死进程是指一个子进程在父进程之前结束,但父进程没有正确地等待(使用
wait
或waitpid
等系统调用)来获取子进程的退出状态。当一个进程结束时,它的退出状态会一直保存,直到父进程通过适当的系统调用来获取它。如果父进程没有获取子进程的退出状态,那么子进程就会变成僵死状态。- 我们的程序在退出的时候:return 0,exit(0),之类的,这个0就是退出码,状态信息,这个东西存储在当前进程的PCB中,会有一个整型值来存储退出码。
- 当我们子进程结束以后,会把退出码写到PCB中,然后希望父进程可以获得到这个退出码,然后父进程就可以看到子进程是正常运行结束还是出错退出。正常的话我们return 0,失败的话我们return -1。
2.代码演示僵死进程
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>
int main(int argc, char *argv[], char *envp)
{
char *s = NULL;
int n = 0;
pid_t pid = fork();
assert(pid != -1);
if (pid == 0)
{
s = "child";
n = 4;
}
else
{
s = "parent";
n = 10;
}
int i = 0;
for (; i < n; i++)
{
printf("pid=%d,s=%s\n", getpid(), s);
}
exit(0);
}
从上图中可以看到,当子进程结束后,并没有消失,仍然可以在系统中观测到,但此时子进程其实已经运行结束了,此时子进程的状态被称为僵死状态,系统把处于该类状态的进程称为僵死进程< defunct >。
3.解决办法
在 Unix/Linux 系统中,可以使用 wait
或 waitpid
函数等待子进程的终止,并获取其退出状态。如果父进程没有执行这些调用,子进程就会一直保留在僵死状态,占用系统资源。
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>
int main(int argc, char *argv[], char *envp)
{
char *s = NULL;
int n = 0;
pid_t pid = fork();
assert(pid != -1);
if (pid == 0)
{
s = "child";
n = 4;
}
else
{
s = "parent";
n = 10;
int val = 0;
pid_t id = wait(&val);
if (WIFEXITED(val))
{
printf("id=%d,val=%d\n", id, WEXITSTATUS(val));
}
}
int i = 0;
for (; i < n; i++)
{
printf("pid=%d,s=%s\n", getpid(), s);
}
exit(0);
}
显而易见,int val = 0; pid_t id = wait(&val); 的执行起到了阻塞作用,因为要获取子进程的退出码,如果子进程还没有结束,就获取不到,父进程就阻塞住了,先将子进程执行完后,获取到退出码了,才开始执行父进程。