我们一直在强调一个概念就是进程是一个程序执行的实例,是内核在虚拟概念下创建的实体,它实例化的体现在用户态就是程序代码和代码使用的变量(存储空间),在内核态就是内核为我们每个进程所保存的数据结构(状态信息)等。

当一个进程由于某种原因停止时,内核并不是直接将为它保存的状态信息从内核移除,相反,进程会一直被保持在一种已经终止的状态,直到被它的父进程回收,当父进程回收已经终止的子进程时,内核会将子进程的退出状态传递给父进程,然后抛弃已经终止的进程,从此刻开始,这个进程才会消失,一个子进程结束但是还没有被父进程回收的进程叫做”僵尸进程”,要是父进程先于子进程结束,那么它的子进程就会称为”孤儿进程”,最终会被PID为1的init进程收养。什么?init进程?好眼熟,对,它是我们引入进程的概念时候提过的。下面我们将介绍父进程等待子进程结束的两个函数,wait()和waitpid()函数。

首先我们来看下man wait的内容:

#include <sys/types.h>
#include <sys/wait.h>
pid_t wait(int *status); //status表示退出状态,指向子进程的退出码
返回:如果成功,返回子进程的ID;如果失败,返回-1。
pid_t waitpid(pid_t pid, int *status, int options); //option可选择设置waitpid不堵塞
返回:如果成功,返回子进程的ID;如果option设置为WNOHANG,则为0;如果其他错误,则为-1。

waitpid()函数:

默认的,当option参数为0的时候,它挂起等待一个子进程的结束;如果子进程中一个进程在刚调用的时候就已经结束了,那么waitpid()就会立即返回,如果正常情况下,这两种都会返回已经终止的进程的PID,并且将这个进程从系统中去除。下面我们再来详细介绍下它的三个参数:

(1):pid:判定等待集合的成员

pid > 0 : 表示等待集合中进程号为pid的特定子进程。

pid = -1 : 表示等待进程的任意一个子进程。

pid = 0 : 表示等待其组ID等于调用进程的组ID的任意一个进程。

(2):status:检查已回收的子进程的退出状态,主要是利用wait.h头文件中的几个宏来检测。

WIFEXITED(status) : 如果子进程通过调用exit或者return正常终止,那么就返回真。

WEXITSTATUS(status) : 返回一个正常终止的子进程的退出状态,返回子进程中exit或_exit的低八位。

WIFSIGNALED(status) : 若子进程异常终止,它就取得一个非零值,表示真。

(3):options : 修改默认行为,介绍最常用的一个。

WNOHANG : 如果等待集合中的任意一个子进程都还没有终止,那么就立即返回,(返回值为0),默认的行为是挂起调用进程等待,直到有子进程结束,这样可以避免阻塞,我们还可以在等待子进程结束的同时干点别的什么事。

下面我们来看一段代码:

#include<stdio.h>
#include<sys/types.h>
#include<sys/wait.h>
#include<errno.h>
#include<unistd.h>
#include<stdlib.h> int main(int argc,char *argv[])
{
pid_t pid;
int status; printf("Hello\n");
pid = fork();
if(pid != 0) {
if(waitpid(-1,&status,0) > 0) {
if(WIFEXITED(status) != 0) {
printf("%d\n",WEXITSTATUS(status));
}
}
}
printf("Bye\n");
exit(100);
}



输出为什么是4行大家懂不懂呢,首先输出Hello没有什么解释的,然后在fork之后我们可以看做有两份一样的代码分别属于父进程和子进程,然后子进程输出Bye,此时的父进程已经被挂起等待子进程的结束,然后子进程100退出码退出,此时父进程结束挂起状态,WEXITSTATUS(status)输出退出码,接着向下执行,再次输出Bye!

错误条件:如果调用进程没有子进程,那么waitpid()函数返回-1,并且设置errno的值为ECHILD;如果waitpid()函数被一个信号中断,那么返回-1,并且设置errno的值为EINTR。举个例子:

#include<stdio.h>
#include<sys/types.h>
#include<sys/wait.h>
#include<errno.h> int main(int argc,char *argv[])
{
pid_t pid;
int status;
pid = waitpid(-1,&status,0); printf("pid is %d\n",pid);
if(errno == ECHILD) {
perror("waitpid");
}
}



我们可以看到对于没有子进程的进程如果调用waitpid()函数的话返回-1,并且将errno值设置为ECHILD,我们可以用perror()函数打印出来错误原因是没有子进程。

wait()函数

介绍完waitpid()函数,wait()函数实际上就不用多说了,它是waitpid()函数的简单版本

wait(status) = waitpid(-1,&status,0)。来段代码练练手:

#include<stdio.h>
#include<sys/wait.h>
#include<sys/types.h>
#include<unistd.h>
#include<stdlib.h> int main(int argc,char *argv[])
{
int pid;
char *msg;
int k;
int exit_code; pid = fork();
switch(pid){
case 0:{
printf("curpid = %d,parentpid = %d\n",pid,getppid());
msg = "children process is running";
k = 5;
exit_code = 55;
break;
}
case -1:{
perror("process creat failed\n");
exit(-1);
}
default:{
printf("childpid is %d ,parentpid is %d\n",pid,getpid());
exit_code = 0;
break;
}
}
printf("------------------------%d\n",pid);
if(pid != 0)
{
int stat_val; // 值为0
int child_pid; child_pid = wait(&stat_val); //wait函数的返回值是终止运行的子进程的pid, 遇到wait函数之后开始执行子进程.
printf("child process has exited,pid = %d\n",child_pid);
if(WIFEXITED(stat_val)){
printf("child exited with code %d\n",WEXITSTATUS(stat_val));
}
else {
printf("child exited abnormally\n");
}
}
//让子进程暂停5秒
else
{
while(k-- > 0)
{
puts(msg);
sleep(1);
}
}
exit(exit_code);
}



我们首先fork一个子进程,然后子进程执行的是case 0 中的代码以及下面else中的代码,父进程执行default中代码和下面if中的代码,但是父进程到wait函数的时候就会挂起来等待子进程的结束。子进程连续输出5次msg的内容之后退出,exit_code=55;这个会被父进程中的 WEXITSTATUS(stat_val) 检测到。

版权声明:本文为博主原创文章,未经博主允许不得转载。

父进程等待子进程结束 waitpid wait的更多相关文章

  1. Linux下利用fork()创建子进程并使父进程等待子进程结束

    int status; pid_t t = fork(); if(t){     waitpid(t, &status, 0); }else{     system("vi temp ...

  2. fork新建进程——父进程等待子进程结束

    #include <sys/types.h>#include<sys/wait.h>#include<unistd.h>#include<stdio.h> ...

  3. linux 进程等待 wait 、 waitpid

    waitpid() 与 wait() 功能相似,都是用户主进程等待子进程结束或中断. 可用于进程之间的同步 wait 函数原型 pid_t wait(int *status); 函数说明 wait() ...

  4. linux系统编程之进程(六):父进程查询子进程的退出,wait,waitpid

    本节目标: 僵进程 SIGCHLD wait waitpid 一,僵尸进程 当一个子进程先于父进程结束运行时,它与其父进程之间的关联还会保持到父进程也正常地结束运行,或者父进程调用了wait才告终止. ...

  5. linux 进程学习笔记-等待子进程结束

    <!--[if !supportLists]-->Ÿ <!--[endif]-->等待子进程结束 pid_t waitpid(pid_t pid, int *stat_loc, ...

  6. Linux 进程--父进程查询子进程的退出状态

    僵尸进程 当一个子进程先于父进程结束运行时,它与其父进程之间的关联还会保持到父进程也正常地结束运行,或者父进程调用了wait才告终止. 子进程退出时,内核将子进程置为僵尸状态,这个进程称为僵尸进程,它 ...

  7. PHP多进程学习(三)__代码案例来了解父进程与子进程的执行顺序

    pcntl_fork创建子进程成功的话,系统就有了2个进程,一个为父进程,一个为子进程,父进程和子进程都继续向下执行,子进程的id号为$pid(父进程会获取子进程的$pid也就是$pid不为0,而子进 ...

  8. 【LINUX】主进程、父进程、子进程、守护进程的概念

    一.摘要 详解父进程.子进程.守护进程的区别,例子稍候补充 二.定义区别 主进程 程序执行的入口,可以理解为常用的main 函数 父进程 对于子进程而言, 子进程的创造者,可有多个子进程. 任何进程都 ...

  9. swoole父进程和子进程之间通信的例子

    <?php /** 这是一个swoole父进程和子进程之间通信的例子 */ //进程创建成功后回调处理 function handle(swoole_process $worker){ //从进 ...

随机推荐

  1. Mesos

    1. 软件定义数据中心 Mesos的二级调度机制: maseos协调每个节点的slave,获取每个节点的机器资源.获取资源后,在相应节点运行framework,在容器中执行任务.从而使得多种类型的服务 ...

  2. HBase内部操作日志说明

    版本:0.94-cdh4.2.1 1. Split Region [regionserver60020-splits-1397585864985] INFO org.apache.hadoop.hba ...

  3. HttpWebRequest-header设置

    http://www.cnblogs.com/yczz/archive/2012/06/01/2530484.html http://blog.csdn.net/htsnoopy/article/de ...

  4. maven多项目配置实践

    工具:IntelliJ IDEA 1.新建maven项目top 略 提示:不使用任何模版 2.在上面的maven项目的目录下(虽然这不一定是必须的),新建另一个maven项目secend. 同上 3. ...

  5. eworkflow工作流系统在iis中发布

    eworkflow工作流系统在iis中发布 win7下面的iis发布eworkflow工作流系统,要带虚拟目录的,如发布成http://localhost/eworkflow/login.aspx这样 ...

  6. 关于intellij IDEA 上junit的用法

    话说,最近正在看视频学java.里面有个叫做junit的东西很有用.但是实话说我摆弄了半天都没弄明白. 今天呢通过一些资料,终于弄清楚了junit的大致用法,这里写出来,用以分享和备忘. 首先,环境和 ...

  7. SQLServer 列出每个表的列和属性

    USE DBProjectSY GO SELECT OBJECT_SCHEMA_NAME(T.[object_id], DB_ID()) AS [架构名] , T.[name] AS [表名] , A ...

  8. Angualr2 - 使用 VS2015

    使用 Vs 2015 快速上手 Angular2 许多程序员希望能够使用 VS2015 作为 IDE 来开发 Angular 2 应用,这里介绍如何设置 VS2015,使用快速上手文件来搭建 ASP. ...

  9. Unsupported major.minor version 52.0

    jdk版本错误,默认应该选择高版本的,请检查你的配置文件引用 java.exe的路径,把它改为绝对路径 分析:其他软件如oracle安装可能会自带一些低版本的jdk,然后你项目引用如果 path=ja ...

  10. gulp使用笔记

    gulp简介 gulp 是基于 Nodejs 的自动任务运行器,能自动化地完成javascript/coffee/sass/less/html/image/css等文件的的测试.检查.合并.压缩.格式 ...