目录
- 僵死进程是指一个子进程在父进程之前结束,但父进程没有正确地等待(使用
wait
或waitpid
等系统调用)来获取子进程的退出状态。当一个进程结束时,它的退出状态会一直保存,直到父进程通过适当的系统调用来获取它。如果父进程没有获取子进程的退出状态,那么子进程就会变成僵死状态。- 我们的程序在退出的时候:return 0,exit(0),之类的,这个0就是退出码,状态信息,这个东西存储在当前进程的PCB中,会有一个整型值来存储退出码。
- 当我们子进程结束以后,会把退出码写到PCB中,然后希望父进程可以获得到这个退出码,然后父进程就可以看到子进程是正常运行结束还是出错退出。正常的话我们return 0,失败的话我们return -1。
- #include
- #include
- #include
- #include
- #include
-
- 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 >。
在 Unix/Linux 系统中,可以使用 wait
或 waitpid
函数等待子进程的终止,并获取其退出状态。如果父进程没有执行这些调用,子进程就会一直保留在僵死状态,占用系统资源。
- #include
- #include
- #include
- #include
- #include
-
- 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); 的执行起到了阻塞作用,因为要获取子进程的退出码,如果子进程还没有结束,就获取不到,父进程就阻塞住了,先将子进程执行完后,获取到退出码了,才开始执行父进程。