首先我们来了解一下所谓的僵尸进程,僵尸进程就是两个进程,一个父进程,一个子进程,其子进程终止后,0-3G的用户内存被回收,而3-4G的部分内存被回收,但是3-4G内存中的PCB等待父进程回收,若PCB未被父进程回收,我们称这个进程为僵尸进程,注,之所以保留PCB,是因为其中保存着子进程的终止状态父进程回收子进程的资源所用到的函数就是我们今天所要说的wait()和waitpid()函数那么我们先来了解一下wait函数wait是一个阻塞函数,如果没有可以回收的子进程,则为阻塞状态 如果无子进程,则返回-1 如果回收成功,则返回子进程的pid返回值pid_twait() 只有一个参数 可以为NULL,也可以为int* status所需要的头文件#include#include
而waitpid()是对wait()函数的优化,我们在父进程使用wait()函数时,因为这个函数是处于阻塞状态的,使父进程不能处理其他事情,这样便浪费了父进程的资源,所以我们引出了waitpid()waitpid()非阻塞 函数有三个参数参数1:<-1时 回收组ID(该组可为用户指定) -1 时 回收任意的相关进程 0 时,回收调用者组中的子进程 >0时,回收一个指定进程参数2:可为NULL也可为int* status 为子进程的终止情况(退出码)参数3:WNOHANG 为非阻塞状态返回值 -1:失败或者没有回收子进程返回-1 >0: 返回回收的子进程ID 0:代表子进程未结束,非阻塞轮询返回注:参数2为传出参数 使用宏函数 WIFEXITED(status) WEXITSTATUS(status) WIFSIGNALED(status) WTERMSIG(status)前两个宏函数为一组 ,后两个宏函数为一组,前两个是正常退出,后两个代表信号退出
下面我们来看一下代码:#include
#include
#include
#include
#include
#include
int main(void)
{
pid_t pid,wpid;
int status,i;//子进程的终止
for(i=0;i<10;i++){
pid = fork();
if(pid == 0)
break;
}
if(pid > 0){
printf("parent pid=%d
",getpid());
//wait阻塞函数,如果没有可回收子进程阻塞,如果没有子进程返回-1,如果回收成功返回子进程pid
//waipid return value:-1,>0,0
//-1:没有可回收子进程
//>0:返回回收子进程pid
//0:代表子进程未结束,非阻塞轮询返回
while((wpid = waitpid(-1,&status,WNOHANG))!=-1){
if(wpid > 0)
{
if(WIFEXITED(status))
printf("parent wait child pid:%d exit code:%d
",wpid,WEXITSTATUS(status));
if(WIFSIGNALED(status))
printf("parent wait child pid:%d signal No:%d
",wpid,WTERMSIG(status));
}
}
}else if(pid == 0){
printf("child id=%d pid=%d
",i,getpid());
if(i<9){
sleep(i);
exit(i);
}
while(1);
}else{
perror("Fork Call Failed:");
exit(0);
}
return 0;
}