[8 进程控制]使用wait函数获取子进程终止状态

1 wait/waitpid介绍

当一个进程正常或异常终止时,内核会向其父进程发送SIGCHLD信号。因为子进程终止是异步的,所以信号发送也是异步的。父进程可以选择忽略该信号,或者提供信号处理函数进行处理。对于SIGCHLD信号,系统默认动作是忽略它。
当父进程调用wait或waitpid时会发生什么:
(1)如果其所有子进程都还在运行,则阻塞。
(2)如果一个子进程已终止,正等待父进程获取其终止状态,则取得该子进程的终止状态后立即返回。
(3)如果其没有任何子进程,则立即出错返回。

#include <sys/wait.h>
// 返回值:若成功,返回子进程ID;若失败,返回0或0-1
pid_t wait(int *statloc);
pid_t waitpid(pid_t pid, int *statloc, int options);

1.1 wait

在子进程终止前,wait使父进程阻塞,而waitpid有一选项,可使父进程不阻塞。
参数statloc存放终止进程的终止状态,如果不关心终止状态,可以置为空指针。
有4个互斥的宏用来取得进程终止的原因,它们名字以WIF开始。

说明

WIFEXITED(status)

若为正常终止子进程返回的状态,则为真。

WIFSIGNALED(status)

若为异常终止子进程返回的状态,则为真。对于这种情况,使用WTERMSIG(status)来获取使子进程终止的信号。

WIFSTOPPED(status)

若为暂停子进程返回的状态,则为真。对于这种情况,使用WSTOPSIG(status)来获取使子进程暂停的信号。

1.2 waitpid

如果一个进程有多个子进程,只要一个子进程终止,wait就返回。如果要等待一个特定的进程终止,需要使用waitpid函数。
waitpid函数中pid参数有如下含义:
(1)pid == -1
等待任一子进程,这时waitpid与wait等效。
(2)pid > 0
等待进程ID与pid相等的子进程。
(3)pid ==0
等待组ID等于调用进程组ID的任一子进程。
(4)pid < -1
等待组ID等于pid绝对值的任一子进程。
waitpid函数返回终止子进程的进程ID,并将子进程的终止状态存放在由staloc指向的存储单元。
options参数说明:

常量

说明

WNOHANG

若由pid指定的子进程并不是立即可用的,则waitpid不阻塞,此时返回值为0

总结,waitpid函数相比wait函数:
(1)waitpid函数可等待一个特定的子进程终止
(2)waitpid提供了一个wait的非阻塞版本

2 使用wait函数获取子进程终止状态

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

void pr_exit(int status)
{
    if (WIFEXITED(status))
        printf("normal termination, exit status = %d\n",
                WEXITSTATUS(status));
    else if (WIFSIGNALED(status))
        printf("abnormal termination, signal number = %d%s\n",
                WTERMSIG(status),
#ifdef    WCOREDUMP
                WCOREDUMP(status) ? " (core file generated)" : "");
#else
                "");
#endif
    else if (WIFSTOPPED(status))
        printf("child stopped, signal number = %d\n",
                WSTOPSIG(status));
}

int main(void)
{
    pid_t    pid;
    int        status;

    // (1)子进程正常终止
    if ((pid = fork()) < 0)
        printf("fork error");
    else if (pid == 0)                /* child */
        exit(7);

    if (wait(&status) != pid)        /* wait for child */
        printf("wait error");
    pr_exit(status);                /* and print its status */

    // (2)子进程异常终止 SIGABRT
    if ((pid = fork()) < 0)
        printf("fork error");
    else if (pid == 0)                /* child */
        abort();                    /* generates SIGABRT */

    if (wait(&status) != pid)        /* wait for child */
        printf("wait error");
    pr_exit(status);                /* and print its status */

    // (3)子进程异常终止 SIGFPE
    if ((pid = fork()) < 0)
        printf("fork error");
    else if (pid == 0)                /* child */
        status /= 0;                /* divide by 0 generates SIGFPE */

    if (wait(&status) != pid)        /* wait for child */
        printf("wait error");
    pr_exit(status);                /* and print its status */

    exit(0);
}

输出:

normal termination, exit status = 7
abnormal termination, signal number = 6 (core file generated)
abnormal termination, signal number = 8 (core file generated)

SIGABRT:6,SIGFPE:8。

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
waitpid函数可以用于等待一个特定的进程或等待任何一个进程。如果使用waitpid等待一个特定的进程,需要传递该进程的PID作为参数。如果使用waitpid等待任何一个进程,则只需要将pid参数设置为-1即可。 以下是一个使用waitpid函数等待进程状态的示例代码: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/wait.h> int main() { pid_t pid = fork(); if (pid == 0) { // 进程 printf("进程正在运行...\n"); sleep(5); printf("进程结束\n"); exit(0); } else if (pid > 0) { // 父进程 int status; pid_t child_pid; do { child_pid = waitpid(pid, &status, WUNTRACED | WCONTINUED); if (child_pid == -1) { perror("waitpid"); exit(EXIT_FAILURE); } if (WIFEXITED(status)) { printf("进程 %d 正常结束,退出状态码 %d\n", child_pid, WEXITSTATUS(status)); } else if (WIFSIGNALED(status)) { printf("进程 %d 被某个信号终止终止信号 %d\n", child_pid, WTERMSIG(status)); } else if (WIFSTOPPED(status)) { printf("进程 %d 被某个信号停止,停止信号 %d\n", child_pid, WSTOPSIG(status)); } else if (WIFCONTINUED(status)) { printf("进程 %d 被继续执行\n", child_pid); } } while (!WIFEXITED(status) && !WIFSIGNALED(status)); printf("父进程结束\n"); exit(EXIT_SUCCESS); } else { perror("fork"); exit(EXIT_FAILURE); } } ``` 在上面的代码中,当父进程调用waitpid函数等待进程时,它会不断轮询进程状态,直到进程正常结束、被信号终止或被信号停止。根据不同的状态,父进程会打印不同的信息。当进程结束后,父进程也会结束。 注意,waitpid函数会阻塞父进程,直到进程状态发生改变。如果父进程不希望阻塞,可以将options参数设置为WNOHANG,这样waitpid函数会立即返回,如果没有任何进程状态发生改变,返回值为0。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值