Linux

C语言-系统-wait()和waitpid()函数

2017-02-08  本文已影响1742人  一叶之界

1. wait()函数

头文件:#include <sys/types.h>/<sys/wait.h>
函数的一般形式:

      pid_t wait(int *status);

参数设置:

wait()会暂时停止目前进程的执行, 直到有信号来到或子进程结束. 如果在调用wait()时子进程已经结束, 则wait()会立即返回子进程结束状态值. 子进程的结束状态值会由参数status 返回, 而子进程的进程识别码也会一快返回. 如果不在意结束状态值, 则参数 status 可以设成NULL.

返回值:如果执行成功则返回子进程识别码(PID), 如果有错误发生则返回-1. 失败原因存于errno 中。
错误代码:

ECHILD:没有等待的子进程;
EINTR:未抓住信号。或者该信号未设置,或者为找到该信号;
// 案例:
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
main()
{
    pid_t pid;
    int status, i;
    if(fork() == 0)
    {
        printf("This is the child process. pid =%d\n", getpid());
        exit(5);
     }
    else
    {
        sleep(1);
        printf("This is the parent process, wait for child...\n";
        pid = wait(&status);
        i = WEXITSTATUS(status);
        printf("child's pid =%d . exit status=%d\n", pid, i);
    }
}

2. waitpid()函数

头文件:#include <sys/types.h>/<sys/wait.h>
函数的一般形式:

      pid_t waitpid(pid_t pid, int *status, int options);

参数设置:

  waitpid()会暂时停止目前进程的执行, 直到有信号来到或子进程结束. 如果在调用wait()时子进程
已经结束, 则wait()会立即返回子进程结束状态值. 子进程的结束状态值会由参数status 返回, 而
子进程的进程识别码也会一快返回. 如果不在意结束状态值, 则参数status 可以设成NULL. 参数pid 
为欲等待的子进程识别码, 其他数值意义如下:
1、pid<-1 等待进程组识别码为pid绝对值的任何子进程.
2、pid=-1 等待任何子进程, 相当于wait().
3、pid=0 等待进程组识别码与目前进程相同的任何子进程.
4、pid>0 等待任何子进程识别码为pid 的子进程.
参数option 可以为0 或下面的OR 组合:
1、WNOHANG:如果没有任何已经结束的子进程则马上返回, 不予以等待;
2、WUNTRACED:如果子进程进入暂停执行情况则马上返回, 但结束状态不予以理会. 子进程的结束状态返回后存于status, 底下有几个宏可判别结束情况;
3、WIFEXITED(status):如果子进程正常结束则为非0 值;
4、WEXITSTATUS(status):取得子进程exit()返回的结束代码, 一般会先用WIFEXITED 来判断是否正常结束才能使用此宏;
5、WIFSIGNALED(status):如果子进程是因为信号而结束则此宏值为真;
6、WTERMSIG(status):取得子进程因信号而中止的信号代码, 一般会先用WIFSIGNALED 来判断后才使用此宏;
7、WIFSTOPPED(status):如果子进程处于暂停执行情况则此宏值为真. 一般只有使用WUNTRACED时才会有此情况;
8、WSTOPSIG(status):取得引发子进程暂停的信号代码, 一般会先用;
9、WIFSTOPPED 来判断后才使用此宏。

返回值:如果执行成功则返回子进程识别码(PID), 如果有错误发生则返回-1. 失败原因存于errno 中。
错误代码:

ECHILD:没有可等待的pid号;
EINTR:未抓住信号。或者该信号未设置,或者为找到该信号
EINVAL:options未被初始化
// 案例:
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
    pid_t pid;
    pid = fork();
    if (pid < 0) 
    {
        perror("fork failed");
        exit(1);
    }
    if (pid == 0) 
    {  //子进程
        int i;
        for (i = 3; i > 0; i--) 
        {
            printf("This is the child\n");
            sleep(1);
        }
        exit(3);
    } 
    else 
    {   //父进程
        int stat_val;
        waitpid(pid, &stat_val, 0);  /*阻塞等待子进程*/
        if (WIFEXITED(stat_val))
        {
            printf("Child exited with code %d\n", WEXITSTATUS(stat_val));
        }
        else if (WIFSIGNALED(stat_val))
        {
            printf("Child terminated abnormally, signal %d\n", WTERMSIG(stat_val));
        }
    }
    return 0;
}

3. wait() 和waitpid() 的区别

wait等待第一个终止的子进程,而waitpid可以通过pid参数指定等待哪一个子进程。当pid=-1、option=0时,waitpid函数等同于wait,可以把wait看作waitpid实现的特例。

waitpid函数提供了wait函数没有提供的三个功能:
1、waitpid等待一个特定的进程,而wait则返回任一终止子进程的状态 。
2、waitpid提供了一个 wait的非阻塞版本,有时希望取得一个子进程的状态, 但不想进程阻塞。
3、waitpid支持作业控制。

上一篇下一篇

猜你喜欢

热点阅读