原文网址:http://blog.csdn.net/jifengszf/article/details/3067841

【waitpid系统调用】   
    
功能描述: 
等待进程改变其状态。所有下面哪些调用都被用于等待子进程状态的改 变,获取状态已改变的子进程信息。状态改变可被认为是:1.子进程已终止。2.信号导致子进程停止执行。3.信号恢复子进程的执行。在子进程终止的情况 下,wait调用将允许系统释放与子进程关联的资源。如果不执行wait,终止了的子进程会停留在"zombie"状态。

如果发现子进程改变了状态,这些调用会立即返回。反之,调用会被阻塞直到子进程状态改变,或者由信号处理句柄所中断(假如系统调用没有通过sigaction的SA_RESTART标志重启动)。

wait 系统调用挂起当前执行中的进程,直到它的一个子进程终止。waitpid挂起当前进程的执行,直到指定的子进程状态发生变化。默认,waitpid只等待 终止状态的子进程,但这种行为可通过选项来改变。waitid系统调用对于等待哪个子进程状态改变提供了更精确的控制。
  
子进程已终 止,父进程尚未对其执行wait操作,子进程会转入“僵死”状态。内核为“僵死”状态的进程保留最少的信息量(进程标识,终止状态,资源使用信息),过后 父进程执行wait时可以获取子进程信息。只要僵死的进程不通过wait从系统中移去,它将会占据内核进程表中的一个栏位。如果进程表被填满,内核将不能 再产生新进程。如果父进程已终止,它的僵死子进程将由init进程收养,并自动执行wait将它们移去。

用法:  
#include <sys/types.h>
#include <sys/wait.h>

pid_t wait(int *status);
pid_t waitpid(pid_t pid, int *status, int options);
int waitid(idtype_t idtype, id_t id, siginfo_t *infop, int options);

参数:   
pid:可能值有以下

小于-1 //意味着等待所有其进程组标识等于pid绝对值的子进程。
 -1 //意味着等待任何子进程。
 0 //意味着等待任何其组标识等于调用进程组标识的进程。
大于0 //意味着等待其进程标识等于pid的进程。

status:指向子进程的返回状态,可通过以下宏进行检索

WIFEXITED(status) //返回真如果子进程正常终止,例如:通过调用exit(),_exit(),或者从main()的return语句返回。
WEXITSTATUS(status) //返回子进程的退出状态。这应来自子进程调用exit()或_exit()时指定的参数,或者来自main内部return语句参数的最低字节。只有WIFEXITED返回真时,才应该使用。
WIFSIGNALED(status) //返回真如果子进程由信号所终止
WTERMSIG(status) //返回导致子进程终止的信号数量。只有WIFSIGNALED返回真时,才应该使用。
WCOREDUMP(status) //返回真如果子进程导致内核转存。只有WIFSIGNALED返回真时,才应该使用。并非所有平台都支持这个宏,使用时应放在#ifdef WCOREDUMP ... #endif内部。
WIFSTOPPED(status) //返回真如果信号导致子进程停止执行。
WSTOPSIG(status) //返回导致子进程停止执行的信号数量。只有WIFSTOPPED返回真时,才应该使用。
WIFCONTINUED(status) //返回真如果信号导致子进程继续执行。

options:可以是0个或多个以下符号常量通过or运算的组合体

WNOHANG //如果没有子进程退出,立即返回
WUNTRACED //如果有处于停止状态的进程将导致调用返回。
WCONTINUED //如果停止了的进程由于SIGCONT信号的到来而继续运行,调用将返回。

下面是Linux特有的选项,不能用于waitid
__WCLONE //只等待"clone"的子进程。一个"clone" 进程即是终止时不会给父进程发送信号,或者不会给父进程发送SIGCHLD信号的进程。
__WALL //等待所有类型的子进程,包括"clone"和"non-clone"。
__WNOTHREAD //不会等待同一线程组的其它线程的子孙。

WUNTRACED 和 WCONTINUED 只有在SIGCHLD信号没有设置SA_NOCLDSTOP标志时才起作用。

idtype,id:这两个参数结合在一起指出应选择等待哪些子进程,可能情况有

idtype == P_PID //等待进程标识与id匹配的子进程。
idtype == P_PGID //等待进程组标识与id匹配的任何子进程。
idtype == P_ALL //等待任何子进程,id无作用

可在options参数中指定的感兴趣的子进程状态改变标志有以下常量,可以通过or运算加以组合

WEXITED //等待已终止的子进程。
WSTOPPED //等待由于信号已停止执行的子进程。
WCONTINUED //等待由于信号已恢复执行的子进程。
WNOHANG //作用如同waitpid。
WNOWAIT //保留子进程的可等待状态,后面的wait调用可再次获取子进程的状态信息。

infop:成功执行返回时,waitid将填充infop所指向结构体的如下字段

si_pid //子进程标识。
si_uid //子进程的真实用户标识。
si_signo //总被设置为SIGCHLD。
si_status //子进程的退出状态,或者导致子进程退出,停止执行或恢复执行的信号,需要根据si_code字段来解释。
si_code //可能值有CLD_EXITED(子进程调用_exit退出), CLD_KILLED(子进程被信号杀死),CLD_STOPPED(信号导致子进程停止执行),CLD_CONTINUED(信号恢复子进程继续执行)。
    
返回说明:   
wait():成功执行时,返回终止子进程的标识。失败返回-1;
waitpid():成功执行时,返回状态改变的子进程标识。失败返回-1;如果指定WNOHANG标志,同时pid指定的进程状态没有发生变化,将返回0。
waitid():成功执行或者WNOHANG标志被设置而id指定的子进程状态没有发生变化时返回0。失败返回-1。

出错值可能有下面这些
ECHILD:参数指定的进程不存在,或者并非调用进程的子进程
EINTR;WNOHANG不被设置,同时捕捉到一个不被阻塞的信号或SIGCHLD信号
EINVAL:options参数无效

例子:

$ ./a.out &
Child PID is 32360
[1] 32359
$ kill -STOP 32360
stopped by signal 19
$ kill -CONT 32360
continued
$ kill -TERM 32360
killed by signal 15
[1]+  Done                    ./a.out
$

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

int main(int argc, char *argv[])
{
    pid_t cpid, w;
    int status;

cpid = fork();
    if (cpid == -1) { perror("fork"); exit(EXIT_FAILURE); }

if (cpid == 0) {            /* 子进程执行 */
        printf("Child PID is %ld/n", (long) getpid());
  if (argc == 1)
            pause();                    /* Wait for signals */
        _exit(atoi(argv[1]));

} else {                    /* 父进程执行 */
        do {
            w = waitpid(cpid, &status, WUNTRACED | WCONTINUED);
            if (w == -1) { perror("waitpid"); exit(EXIT_FAILURE); }

if (WIFEXITED(status)) {
                printf("exited, status=%d/n", WEXITSTATUS(status));
            } else if (WIFSIGNALED(status)) {
                printf("killed by signal %d/n", WTERMSIG(status));
            } else if (WIFSTOPPED(status)) {
                printf("stopped by signal %d/n", WSTOPSIG(status));
            } else if (WIFCONTINUED(status)) {
                printf("continued/n");
            }
        } while (!WIFEXITED(status) && !WIFSIGNALED(status));
        exit(EXIT_SUCCESS);
    }

【转】linux : waitpid函数的更多相关文章

  1. linux多进/线程编程(3)——wait、waitpid函数和孤儿、僵尸进程

    当使用fork创建多个进程后,需要解决子进程回收的问题.wait和waitpid函数就是做这个工作的. 假设子进程没有合理的回收,可能会带来两个问题: 1.孤儿进程(父进程挂了,子进程活着),孤儿进程 ...

  2. 【Linux】僵尸进程,孤儿进程以及wait函数,waitpid函数(有样例,分析很详细)

    本文内容: 1.僵尸进程,孤儿进程的定义,区别,产生原因,处理方法 2.wait函数,waitpid函数的分析,以及比较 背景:由于子进程的结束和父进程的运行是一个异步的过程,即父进程永远无法预测子进 ...

  3. waitpid 函数详解

    关于Linux中waitpid函数的一些使用说明: #include<sys/types.h> #include<sys/wait.h> 定义函数 pid_t waitpid( ...

  4. linux system函数引发的错误

    转: https://my.oschina.net/renhc/blog/54582 先看一下问题 简单封装了一下system()函数:   int pox_system(const char *cm ...

  5. Linux Clone函数

    Linux Clone函数 之前某一次有过一次面试,问了内核中是怎么创建命名空间的? 下面就来扒一扒clone的精髓,以及如何通过它创建命名空间. 目录 Linux Clone函数 使用clone创建 ...

  6. Linux wait函数详解

    wait和waitpid出现的原因 SIGCHLD --当子进程退出的时候,内核会向父进程SIGCHLD信号,子进程的退出是个异步事件(子进程可以在父进程运行的任何时刻终止) --子进程退出时,内核将 ...

  7. Linux open函数

    Linux open函数 open 函数用于打开和创建文件.以下是 open 函数的简单描述 #include <fcntl.h> int open(const char *pathnam ...

  8. 进程控制之wait和waitpid函数

    当一个进程正常或异常终止时,内核就向其父进程发送SIGCHLD信号.因为子进程终止是个异步事件(这可以在父进程运行的任何时候发生),所以这种信号也是内核向父进程发的异步通知.父进程可以选择忽略该信号, ...

  9. linux内核函数库文件的寻找

    linux内核函数的so库文件怎么找呢? 首先还是要产生一个进程的coredump文件的 linux有一个lib-gdb.so库,这个进程的coredump文件中所有load段的最后一个load段中, ...

随机推荐

  1. 循环队列实现(C++) Ring Buffer

    循环队列:队列有着先入先出的特性.但是对于队列如果删除队头以后剩下的空间将不会被释放,又由于队列只能由队尾插入这就导致被删除部分的空间被浪费.解决这个问题就是循环队列.循环队列顾名思义就是将队列串起来 ...

  2. sigleSchool 存储过程例1

    CREATE OR REPLACE PROCEDURE SINGLSCHOOL( PICIID IN VARCHAR2, SCHOOLID IN NUMBER, SCHETYPE IN number, ...

  3. java核心技术记录之java术语

    术语名 缩写 解释 Java Development Kit JDK 编写java程序的程序员使用的软件 Java Runtime Environment JRE 运行java程序的用户使用的软件 S ...

  4. JsRender系列demo-对null 和boolen类型数据的探讨

    废话不说了,直接上代码 <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <he ...

  5. ZOJ 3791 An Easy Game(DP)

    题目链接 题意 : 给你两个长度为N的字符串,将第一个字符串每次只能变化M个,问变换K次之后变成第二个字符串一共有几种方法. 思路 : DP.dp[i][j]表示变了 i 次之后有j个不一样的字母的方 ...

  6. 简单易懂的现代魔法——Play Framework攻略3

    接前文:http://www.cnblogs.com/Kassadin/p/4343682.html 目前为止,我们已经研究了Play Framework的体系结构以及Web应用程序的工作原理.本来今 ...

  7. MySQL数据导出导入【转】

    MySQL基础 关于MySQL数据导出导入的文章,目的有二: 1.备忘 2.供开发人员测试 工具 mysqlmysqldump 应用举例 导出 导出全库备份到本地的目录 mysqldump -u$US ...

  8. JAVA:23种设计模式详解(转)2

    我们接着讨论设计模式,上篇文章我讲完了5种创建型模式,这章开始,我将讲下7种结构型模式:适配器模式.装饰模式.代理模式.外观模式.桥接模式.组合模式.享元模式.其中对象的适配器模式是各种模式的起源,我 ...

  9. C++:基类和派生类

    4.1 派生类的声明 继承实例如下: class Person{ //声明基类Person public: void print() { cout<<"name:"&l ...

  10. iOS:自定义工具栏、导航栏、标签栏

    工具栏为UIToolBar,导航栏UINavigationBar,标签栏UITabBar.它们的样式基本上时差不多的,唯一的一点区别就是,工具栏一般需要自己去创建,然后添加到视图中,而导航栏和标签栏不 ...