父进程等待子进程的退出
收集子进程的退出状态
参考文章链接
wait()、waitpid() 及 waitid() 介绍
子进程退出状态不被收集的话,会变成僵尸进程(僵死进程,Z)
比如说,之前的代码里
vfork() 以后,用 exit() 结束子进程,打开 ps -aux|grep a.out 会发现,子进程仍然存在,不会消失,
#include<sys/types.h>
#include<sys/wait.h>
pid_t wait (int * status);
pid_t waitpid (pid_t pid, int * stastus, int optitions);
int waitid (idtype_t idtype, id_t id, siginfo_t * infop , int options );
关于 pid_t wait(int * status)
进程一旦调用了wait,就立即阻塞自己,由wait自动分析是否当前进程的某个子进程已经退出,如果让它找到了这样一个已经变成僵尸的子进程,wait就会收集这个子进程的信息,并把它彻底销毁后返回;如果没有找到这样一个子进程,wait就会一直阻塞在这里,直到有一个出现为止。
参数:
status用来保存被收集进程退出时的一些状态,它是一个指向int类型的指针。但如果我们对这个子进程是如何死掉的毫不在意,只想把这个僵尸进程消灭掉,(事实上绝大多数情况下,我们都会这样想),我们就可以设定这个参数为NULL,就象下面这样:
pid = wait(NULL);
返回值:
如果成功,wait会返回被收集的子进程的进程ID
如果调用进程没有子进程,调用就会失败,此时wait返回-1,同时errno被置为ECHILD。
pid_t waitpid(pid_t pid,int *status,int options)
从本质上讲,系统调用waitpid和wait的作用是完全相同的,但waitpid多出了两个可由用户控制的参数pid和options,从而为我们编程提供了另一种更灵活的方式。
参数:(status同上)
pid:从参数的名字pid和类型pid_t中就可以看出,这里需要的是一个进程ID。但当pid取不同的值时,在这里有不同的意义。
pid>0时,只等待进程ID等于pid的子进程,不管其它已经有多少子进程运行结束退出了,只要指定的子进程还没有结束,waitpid就会一直等下去。
pid=-1时,等待任何一个子进程退出,没有任何限制,此时waitpid和wait的作用一模一样。
pid=0时,等待同一个进程组中的任何子进程,如果子进程已经加入了别的进程组,waitpid不会对它做任何理睬。
pid<-1时,等待一个指定进程组中的任何子进程,这个进程组的ID等于pid的绝对值。
options:
options: options提供了一些额外的选项来控制waitpid,目前在Linux中只支持WNOHANG和WUNTRACED两个选项,这是两个常数,可以用"|"运算符把它们连接起来使用。
比如:
ret=waitpid(-1,NULL,WNOHANG | WUNTRACED);
如果我们不想使用它们,也可以把options设为0,如:ret=waitpid(-1,NULL,0);
WNOHANG :若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若结束,则返回该子进程的ID。
WUNTRACED : 若子进程进入暂停状态,则马上返回,但子进程的结束状态不予以理会。WIFSTOPPED(status)宏确定返回值是否对应与一个暂停子进程。
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int main(){
pid_t pid;
int cnt = 0;
int status = 0;
pid = fork();
if( pid > 0){
wait(&status); // wait(NULL); // 对于 wait 函数,如果 参数为 NULL;那便不接收子进程退出时的返回值,如果 设置了一个地址给 wait 那个便将 子进程的返回值 放入此地址
// 这样的代码,会发现,子进程先跑完,为什么,好好的理解一下,什么叫阻塞,就是暂停啊!!
while(1){
printf("this is a father, pid = %d; cnt = %d\n",getpid(),cnt);
sleep(1); // 令进程在这停顿 1S
}
}
else if (pid == 0){
while(1){
cnt++;
printf("this is a son, pid = %d\n",getpid());
if(cnt > 3){
exit(90);
}
}
}
// 但是,直接打印 status不可取
printf("status = %d\n",status);
// 这样根本得不到 exit(4); 的退出值 4,printf("%d",status); 的结果是一个奇怪的值;
// 想要打印出 4 必须要使用 wait 返回的终止状态的宏,这里可以用 printf("%d",WEXITSTATUS(status));
// 但是,我们可以使用 WIFEXITED(status); 来判断其是否正常退出,若其返回值非 0 ,则表示其正常结束
return 0;
}