wait和waitpid函数可以用来获得子进程退出的信息,防止子进程成为僵死进程。它们具有如下性质:
- 如果其所有子进程都还在运行,则阻塞。
- 进程终止后成为僵死进程,函数取得终止状态后立即返回。
- 如果调用者没有子进程,则立即出错返回。
两个函数都会返回终止子进程的进程ID。因为一个进程(正常或异常)终止时,内核会向其父进程发送一个SIGCHLD信号,所以令信号处理程序调用wait函数,则wait会立刻返回,防止了僵死进程的出现。
wait和waitpid函数的区别如下:
- 当没有子进程终止时,wait会阻塞,而waitpid可以选择非阻塞调用。
- wait等待任意一个终止的子进程,而waitpid则可以指定需要等待的进程。
这里重点讨论一种应用。当一个进程fork出一个子进程,它不想等待子进程终止,又不希望子进程处于僵死状态直到父进程终止(这是有可能的,因为父进程可能要运行很久),实现这一要求的技巧是调用fork两次。看代码:
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main(void)
{
pid_t pid;
if ((pid = fork()) == 0)
{
// 子进程再fork一个子进程
if ((pid = fork()) > 0)
exit(0); // 第一个子进程退出,第二个子进程由init领养
// 以下是第二个子进程代码
sleep(2);
printf("second child, parent pid = %d\n", getppid());
exit(0);
}
// 等待第一个子进程退出
if (waitpid(pid, NULL, 0) != pid) // waitpid返回终止进程的进程ID
printf("waitpid error!\n");
return 0;
}
运行结果:
父进程收到第一个子进程的退出状态后便终止了,这时第二个子进程还没有终止。但是这不会使第二个子进程称为僵死进程,因为它的父进程现在变为了init进程了。2秒中后第二个子进程终止,由init进程调用wait获取其终止状态,程序正常退出。这也解释了上图中,a.out运行后马上返回到shell中的原因。
参考:
《unix环境高级编程》 P179-P183.