探究wait与waitpid之间的那些事

Posted bwbfight

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了探究wait与waitpid之间的那些事相关的知识,希望对你有一定的参考价值。

首先先说说进程

1.进程产生
操作运行了一个程序,至少会产生一个进程

问题:进程与程序的区别?
回答:程序就是一个编译好的可执行文件,而进程是操作系统为了执行一个程序而创建的,进程是程序的一次执行过程

二 进程相关的命令

1.ps -ef
查看进程的PID和PPID

2.ps aux
查看进程状态 R:运行态 S:可中断的等待态 D:不能中断的等待态 T:停止态 Z:僵尸态(PID资源未释放)

3.kill
<1>kill -l 查看系统中信号
<2>发送信号到指定的进程
kill -信号 PID
例如:
kill -SIGSTOP 5207

4.killall
给同名的进程发信号

killall -信号 进程名

5.查看系统允许的最大进程个数
cat /proc/sys/kernel/pid_max -> 32768

三 创建子进程

pid_t fork(void);
功能:创建子进程
返回值:
成功给父进程返回子进程的PID,给进程返回0,失败返回-1

思考:如何创建子进程?
回答:复制父进程

问题:为什么创建子进程?
回答:创建子进程的目的是让子进程执行独立任务

问题:子进程和父进程会不会相互影响
回答:不会,父子进程的地址空间是独立

 

四 父进程回收僵尸态子进程

首先应该知道的概念:

僵尸进程:进程结束,但是父进程没有对它进行收尸处理,即父进程没有对它做资源的回收处理,在内核进程树上还占用一个节点
孤儿进程:父亲进程结束,子进程就会变成孤儿进程,孤儿进程会自动被init进程收养

pid_t wait(int *status);
功能:等待僵尸态子进程
参数:
@status 获取僵尸态子进程状态
返回值:
成功返回僵尸态子进程PID,失败返回-1(没有子进程)

常用用法: wait(NULL);

pid_t waitpid(pid_t pid, int *status, int options);
参数:
@pid -1:等待任意一个子进程退出 子进程的PID:等待指定PID子进程
@status 获取子进程退出的状态
@options 0:阻塞的方式调用 WNOHANG:非阻塞方式调用
返回值:
成功返回僵尸态子进程PID,失败返回-1, 如果没有子进程处于僵尸态,且非阻塞方式调用,此时立即返回,返回值为0

waitpid(-1,NULL,0); => wait(NULL);
waitpid(pid,NULL,0);

waitpid(-1,NULL,WNOHANG);//常用的用法

waitpid(pid,NULL,WNOHANG);

五 结束一个进程

return : 在main函数使用return,用来结束进程
_exit() : 用来结束一个进程
exit() : 用来结束一个进程,刷新缓存

exit(EXIT_SUCCESS) 成功
exit(EXIT_FAILURE) 失败

 

六:实例如下:

int main(int argc, const char *argv[])
{
  int ret;
  int status;
  nt child_pid1;
  int child_pid2;

  child_pid1 = fork();
  if(child_pid1 < 0){
    perror("Fail to fork");
    return -1;
  }

  if(child_pid1 == 0){
    printf("hello word! ");
    return 0x18;
  }

    if(child_pid1 > 0){

    child_pid2 = fork();
  if(child_pid2 < 0){
    perror("Fail to fork");
    return -1;
  }

  if(child_pid2 == 0){
  while(1)
  ;
  }

  if(child_pid2 > 0){
  while(1)
  {
    printf("wait ... ");
    ret = wait(&status);

    printf("ret : %d ",ret);

    if(ret < 0){
    printf("Fail to wait : %s ",strerror(errno));
    return -1;
     }

  if(WIFEXITED(status)){
    printf("The child process normally exit , return : %#x ",WEXITSTATUS(status));
  }else if(WIFSIGNALED(status)){
    printf("The Child process terminated by signal : %d ",WTERMSIG(status));
     }

       }
     }  
  }

 return 0;
}

这个程序主要功能为,创建两个子进程,父进程分别等待两个子进程退出,一个为正常退出,一个通过发送信号使其退出

 测试结果如下

编译执行:

技术分享图片

 

可以看到子进程1正常退出,用WEXITSTATUS(status)获取了她的返回码

而子进程2一直在轮询,然后,我们查看后台进程

执行 ps aux|grep a.out

技术分享图片

可以看到其中一个子进程的确已经退出,然后我们发送信号让另一个子进程退出(你能确定那个是子进程吗?)

kill -SIGKILL 10611  (当然除了SIGKILL ,还可以被其他信号中断(具体执行kill -l),为什么?)

结果如下:

技术分享图片

可知进程2已经被信号终止,并且通过WTERMSIG获取了是那个信号

既然wait(&status),可以等待子进程退出,如果换成waitpid呢,请看下面的改造:

技术分享图片

测试如下:

技术分享图片

 

可以看出waitpaid(-1,&status,WNOHANGE),会去检测,当前有没有处于僵尸态的子进程,如果没有,立即返回,返回值为0

同理发送信号回终止子进程2

技术分享图片

 其他用法上面已经说过,就不举例说明!





































































以上是关于探究wait与waitpid之间的那些事的主要内容,如果未能解决你的问题,请参考以下文章

TCP协议那些事

二代测序那些事

Java基础__Java中多线程那些事

OO_Unit2 关于性能优化与测试的那些事

Java线程同步那些事

Java线程同步那些事