pipe函数、SIGCHLD、execvp

pipe函数

以下是一个使用C语言编写的通过管道(pipe)进行进程间通信的示例代码:

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

int main() {
    int pipefd[2];
    pid_t pid;
    char buffer[50];

    // 创建管道
    if (pipe(pipefd) == -1) {
        perror("pipe");
        exit(EXIT_FAILURE);
    }

    // 创建子进程
    pid = fork();

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

    // 子进程
    if (pid == 0) {
        close(pipefd[1]);  // 关闭写端

        // 从管道读取数据
        read(pipefd[0], buffer, sizeof(buffer));
        printf("子进程读取到的数据:%s\n", buffer);

        // 关闭读端
        close(pipefd[0]);
        exit(EXIT_SUCCESS);
    }

    // 父进程
    else {
        close(pipefd[0]);  // 关闭读端

        char message[] = "Hello from parent process!";

        // 将数据写入管道
        write(pipefd[1], message, sizeof(message));
        printf("父进程写入数据成功!\n");

        // 关闭写端
        close(pipefd[1]);
        exit(EXIT_SUCCESS);
    }
}

该示例中,首先使用pipe函数创建一个管道,然后调用fork函数创建一个子进程。在子进程中,通过read函数从管道中读取数据,并打印输出。在父进程中,通过write函数将数据写入管道。通过这种方式,子进程和父进程之间就可以进行通信了。请注意,在使用完管道后,需要分别关闭管道的读端和写端。

SIGCHLD,execvp、pipe

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

void sigchld_handler(int signum) {
    int status;
    pid_t pid;

    // 等待子进程退出并获取退出状态
    while ((pid = waitpid(-1, &status, WNOHANG)) > 0) {
        if (WIFEXITED(status)) {
            printf("子进程 %d 正常退出,退出状态码: %d\n", pid, WEXITSTATUS(status));
        } else if (WIFSIGNALED(status)) {
            printf("子进程 %d 被信号终止,终止信号: %d\n", pid, WTERMSIG(status));
        }
    }
}

int main() {
    int pipefds[2];
    pid_t pid;
    int status;
    struct sigaction sa;

    // 创建管道
    if (pipe(pipefds) == -1) {
        perror("pipe");
        exit(1);
    }

    // 注册SIGCHLD信号处理函数
    sa.sa_handler = sigchld_handler;
    sigemptyset(&sa.sa_mask);
    sa.sa_flags = SA_RESTART;
    if (sigaction(SIGCHLD, &sa, NULL) == -1) {
        perror("sigaction");
        exit(1);
    }

    // 创建子进程
    pid = fork();
    if (pid == -1) {
        perror("fork");
        exit(1);
    } else if (pid == 0) {
        // 子进程中将标准输出重定向到管道写端
        close(pipefds[0]); // 关闭管道读端
        dup2(pipefds[1], STDOUT_FILENO); // 将标准输出重定向到管道写端
        close(pipefds[1]); // 关闭管道写端

        // 执行外部命令
        char *args[] = {"ls", "-l", NULL};
        if (execvp(args[0], args) == -1) {
            perror("execvp");
            exit(1);
        }
    } else {
        // 父进程从管道读端读取子进程的输出
        close(pipefds[1]); // 关闭管道写端
        char buffer[1024];
        int nbytes;

        while ((nbytes = read(pipefds[0], buffer, sizeof(buffer))) > 0) {
            write(STDOUT_FILENO, buffer, nbytes);
        }

        // 等待子进程退出
        if (waitpid(pid, &status, 0) == -1) {
            perror("waitpid");
            exit(1);
        }

        // 检查子进程退出状态
        if (WIFEXITED(status)) {
            printf("子进程 %d 正常退出,退出状态码: %d\n", pid, WEXITSTATUS(status));
        } else if (WIFSIGNALED(status)) {
            printf("子进程 %d 被信号终止,终止信号: %d\n", pid, WTERMSIG(status));
        }
    }

    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值