进程等待

进程等待的必要性

  • 之前讲过,子进程退出,父进程如果不管不顾,就可能造成‘僵尸进程’的问题,进而造成内存泄漏。
  • 另外,进程一旦变成僵尸状态,那就刀枪不入,“杀人不眨眼”的kill -9 也无能为力,因为谁也没有办法 杀死一个已经死去的进程。
  • 最后,父进程派给子进程的任务完成的如何,我们需要知道。如,子进程运行完成,结果对还是不对, 或者是否正常退出。
  • 父进程通过进程等待的方式,回收子进程资源,获取子进程退出信息。

进程等待的方法

wait方法
 #include<sys/types.h>
 #include<sys/wait.h>
 
 pid_t wait(int *status);

返回值:
	成功返回被等待进程pid,失败返回-1
参数:
	输出型参数,获取子进程的退出状态,不关心则可以设置为NULL
waitpid方法
pid_t waitpid(pid_t pid, int *status, int options);

返回值:
	正常返回的时候,waitpid返回收集到的子进程ID
	如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0; 
    如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在; 

参数:
	pid:
		pid = -1,等待任一个子进程,与wait等效
		pid > 0,等待其进程ID与pid相等的子进程。 
	status:
		 WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出)        
		 WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码) 
	options:
		 WNOHANG: 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若正常结束,则返回该子进程的ID。 

  • 如果子进程已经退出,调用wait/waitpid时,wait/waitpid会立即返回,并且释放资源,获得子进程退 出信息。
  • 如果在任意时刻调用wait/waitpid,子进程存在且正常运行,则进程可能阻塞。
  • 如果不存在该子进程,则立即出错返回。

获取子进程status

  • wait和waitpid,都有一个status参数,该参数是一个输出型参数,由操作系统填充。
  • 如果传递NULL,表示不关心子进程的退出状态信息。
  • 否则,操作系统会根据该参数,将子进程的退出信息反馈给父进程。
  • status不能简单的当作整形来看待,可以当作位图来看待,具体细节如下图(只研究status低16比特位):

在这里插入图片描述
测试代码:

#include<stdio.h>
#include<sys/wait.h>
#include<string.h>
#include<errno.h>                                                                                      
#include<unistd.h>    
#include<stdlib.h>    
    
int main()    
{    
  pid_t pid;    
  if((pid = fork()) == -1)    
  {    
    perror("fork error");    
    exit(1);    
  }    
  else if(pid == 0)    
  {    
    //child    
    sleep(20);    
    exit(10);    
  }    
  else{    
    //father    
    int st;    
    int ret = wait(&st);    
    
    if(ret > 0 && (st & 0x7F) == 0)    
    {    
      //正常退出    
      printf("child exit code:%d\n", (st >> 8)&0x7F);    
    }    
    else if(ret > 0)    
    {    
      //异常退出    
      printf("sig code:%d\n", st&0x7F);    
    }    
 }
  return 0;
}             

运行结果:

等待30秒退出
在这里插入图片描述
用ps aux查看进程的pid
在这里插入图片描述
然后 kill -9 22391(进程号)杀掉该进程
在这里插入图片描述

具体代码实现
  • 进程的阻塞等待方式
#include<stdio.h>                                                                                      
#include<sys/wait.h>
#include<string.h>
#include<errno.h>
#include<unistd.h>
#include<stdlib.h>

int main()
{
  pid_t pid;
  pid = fork();
  if(pid < 0)
  {
    printf("%s fork error\n",__FUNCTION__);
    return 1;
  }
  else if(pid == 0)
  {
   //child
   printf("child is run, pid is : %d\n", getpid());
   sleep(5);
   exit(256);
  }
  else{
    int status = 0;
    pid_t ret = waitpid(-1, &status, 0);//阻塞式等待,等待55
    printf("this is test for wait\n");
    if(WIFEXITED(status) && ret == pid)          //WIFEXITED(status) 这个宏用来指出子进程是否为正常退出的,如果是,它会返回一个非零值。    
    {    
      printf("wait child 5s success, child return code is : %d.\n", WEXITSTATUS(status));    
    }    
    else{
      printf("wait child failed, return.\n");
      return 1;
    }
  }
  return 0;
}             

上函数中提到两个函数,给大家讲一下

在这里插入图片描述
WEXITSTATUS这个函数是做什么的?

可以获取子进程调传送给exit或用 exit 函数参数的低 8位。

1、WEXITSTATUS 是返回子进程的退出码,用来判断子进程的退出值。当WIFEXITED(status)这个宏用来指出子进程是否为正常退出的,如果是,它会返回一个非零值。
2、WEXITSTATUS(status)取得子进程exit()返回的结束代码,一般会先用WIFEXITED 来判断是否正常结束才能使用此宏。

Linux下 WIFEXITED, WEXITSTATUS的比较

1,WIFEXITED(status) 这个宏用来指出子进程是否为正常退出的,如果是,它会返回一个非零值。

(请注意,虽然名字一样,这里的参数status并不同于wait唯一的参数–指向整数的指针status,而是那个指针所指向的整数,切记不要搞混了。)

2,WEXITSTATUS(status)当WIFEXITED返回非零值时,我们可以用这个宏来提取子进程的返回值,如果子进程调用exit(5)退出,WEXITSTATUS(status)就会返回5;如果子进程调用exit(7),WEXITSTATUS(status)就会返回7。请注意,如果进程不是正常退出的,也就是说,WIFEXITED返回0,这个值就毫无意义。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值