Linux 进程 1.4— 父进程等待子进程退出 wait() 函数处理僵尸进程

父进程等待子进程的退出
收集子进程的退出状态
参考文章链接
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;
}

关于 WIFEXITED 和 WIFSIGNALED

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值