waitpid函数

 作用同于wait,但可指定pid进程清理,可以不阻塞。

pid_t waitpid(pid_t pid,int *status,int options);成功:返回清理掉的子进程ID;失败:-1(无子进程)

特殊参数和返回情况:

参数pid:

       >0 回收指定ID的子进程

       -1 回收任意子进程(相当于wait)

       0 回收和当前调用waitpid一个组的所有子进程

       < -1 回收指定进程组内的任意子进程

返回0:参数3为WNOHANG,且子进程正在运行。

注意:一次wait或waitpid调用只能清理一个子进程,清理多个子进程需要用到循环

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

int main(int argc, char *argv[])
{
    int n = 5, i;                //默认创建5个子进程
    pid_t p, q;

    if(argc == 2){    
        n = atoi(argv[1]);
    }

    for(i = 0; i < n; i++)     {//出口1,父进程专用出口
        p = fork();
        if(p == 0) {
            break;            //出口2,子进程出口,i不自增
        } else if (i == 3){
            q = p;
        }
    }

    if(n == i){
        sleep(n);
        printf("I am parent, pid = %d\n", getpid(), getgid());
        //pid_t pid = waitpid(q, NULL, WNOHANG);
//        pid_t pid = wait(NULL);
        //printf("child pid = %d\n", pid);
        while(1);

    } else {
        sleep(i);
        printf("I'm %dth child, pid = %d, gpid=%d\n", 
                i+1, getpid(), getgid());
        while(1);
    }

    return 0;
}
/***
waitpid.c
***/
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <sys/wait.h>

int main(void)
{
    pid_t pid, pid2, wpid;
    int flg = 0;

    pid = fork();
    pid2 = fork();

    if(pid == -1){
        perror("fork error");
        exit(1);
    } else if(pid == 0){        //son
        printf("I'm process child, pid = %d\n", getpid());
        sleep(5);                
        exit(4);
    } else {                    //parent
        do {
            wpid = waitpid(pid, NULL, WNOHANG);
            //wpid = wait(NULL);
            printf("---wpid = %d--------%d\n", wpid, flg++);
            if(wpid == 0){
                printf("NO child exited\n");
                sleep(1);        
            }
        } while (wpid == 0);        //子进程不可回收

        if(wpid == pid){        //回收了指定子进程
            printf("I'm parent, I catched child process,"
                    "pid = %d\n", wpid);
        } else {
            printf("other...\n");
        }
    }

    return 0;
}
/***
waitpid2.c
***/
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <sys/wait.h>

int main(void)
{
    pid_t pid, pid2, wpid;
    int flg = 0;

    pid = fork();
    pid2 = fork();

    if(pid == -1){
        perror("fork error");
        exit(1);
    } else if(pid == 0){        //son
        printf("I'm process child, pid = %d\n", getpid());
        sleep(5);                
        exit(4);
    } else {                    //parent
        do {
            wpid = waitpid(pid, NULL, WNOHANG);
            //wpid = wait(NULL);
            printf("---wpid = %d--------%d\n", wpid, flg++);
            if(wpid == 0){
                printf("NO child exited\n");
                sleep(1);        
            }
        } while (wpid == 0);        //子进程不可回收

        if(wpid == pid){        //回收了指定子进程
            printf("I'm parent, I catched child process,"
                    "pid = %d\n", wpid);
        } else {
            printf("other...\n");
        }
    }

    return 0;
}
/***
waitpid3.c
***/
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>

int main(int argc, char *argv[])
{
    int n = 5, i;                
pid_t p, q;

    if(argc == 2){    
        n = atoi(argv[1]);
    }
    q = getpid();

    for(i = 0; i < n; i++)     {
        p = fork();
        if(p == 0) {
            break;            
        } 
    }

    if(n == i){  // parent
        sleep(n);
        printf("I am parent, pid = %d\n", getpid());
        for (i = 0; i < n; i++) {
            p = waitpid(0, NULL, WNOHANG);
            printf("wait  pid = %d\n", p);
        }
    } else {
        sleep(i);
        printf("I'm %dth child, pid = %d\n", 
                i+1, getpid());
    }
    return 0;
}

waitpid:

       参1:    pid  > 0       指定进程id回收

                     pid = -1        回收任意子进程

                     pid = 0          回收本组任意子进程

                     pid < -1        回收该进程组的任意子进程

       参2:    status:

                     返回:成功:pid  失败 -1

                     status:传出参数

                     1: 阻塞等待子进程

                     2: 回收子进程资源

                     3:    获取子进程结束状态:1)WIFEXITED()真

                                                                      WEXITSTATUS()获取子进程退出状态

                                                               2)WIFSIGNALED() 真

                                                                      WTERMSIG()获取导致子进程终止的信号的                                                                                           编码

参3:    0 :(wait)阻塞回收

              WBNIOHANG:非阻塞回收(轮询)

返回值:       成功:pid  失败 -1  返回 0 值: 参3传WNOHANG,并且子进程尚未结束。