IPC 之管道

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

void    client(int, int), server(int, int);

int
main(int argc, char **argv)
{
        int             pipe1[2], pipe2[2];
        pid_t   childpid;

        pipe(pipe1);    /* create two pipes */
        pipe(pipe2);

        if ( (childpid = fork()) == 0) {                /* child */
                close(pipe1[1]);
                close(pipe2[0]);

                server(pipe1[0], pipe2[1]);
                exit(0);
        }
                /* 4parent */
        close(pipe1[0]);
        close(pipe2[1]);

        client(pipe2[0], pipe1[1]);

        waitpid(childpid, NULL, 0);             /* wait for child to terminate */
        exit(0);
}

mainpipe.c

// ubuntu16.04 pipe 默认是半双工管道  

man pipe

pipe()  creates  a  pipe,  a unidirectional data channel that can be used for interprocess communication.

 

 

client.c

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>

#define MAXLINE 128

void
client(int readfd, int writefd)
{
        size_t  len;
        ssize_t n;
        char    buff[MAXLINE];

                /* 4read pathname */
        fgets(buff, MAXLINE, stdin);
        len = strlen(buff);             /* fgets() guarantees null byte at end */
        if (buff[len-1] == '\n')
                len--;                          /* delete newline from fgets() */

                /* 4write pathname to IPC channel */
        write(writefd, buff, len);

                /* 4read from IPC, write to standard output */
        while ( (n = read(readfd, buff, MAXLINE)) > 0)
                write(STDOUT_FILENO, buff, n);
}

 

server.c

#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include "my_err.h"

#define MAXLINE 128

void
server(int readfd, int writefd)
{
        int             fd;
        ssize_t n;
        char    buff[MAXLINE+1];

                /* 4read pathname from IPC channel */
        if ( (n = read(readfd, buff, MAXLINE)) == 0)
                err_quit("end-of-file while reading pathname");
        buff[n] = '\0';         /* null terminate pathname */

        if ( (fd = open(buff, O_RDONLY)) < 0) {
                        /* 4error: must tell client */
                snprintf(buff + n, sizeof(buff) - n, ": can't open, %s\n",
                                 strerror(errno));
                n = strlen(buff);
                write(writefd, buff, n);

        } else {
                        /* 4open succeeded: copy file to IPC channel */
                while ( (n = read(fd, buff, MAXLINE)) > 0)
                        write(writefd, buff, n);
                close(fd);
        }
}

gcc -o mainpipe mainpipe.c client.c server.c

./mainpipe

/etc/issue

Ubuntu 16.04.5 LTS \n \l

 

./mainpipe

/no/such/file

/no/such/file: can't open, No such file or directory

 

popen 与 pclose

popen 创建一个管道并启动一个进程,改进程要么从管道读出标准输入,要么往管道写入标准输出。

FILE *popen(const char *command, const char *type);

int pclose(FILE *stream);

type is "r", calling process read the stdout of command

type is "w", calling process write to the stdin of the command

客户-服务器程序

#include "unistd.h"
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#define MAXLINE 128

int
main(int argc, char **argv)
{
        size_t  n;
        char    buff[MAXLINE], command[MAXLINE];
        FILE    *fp;

                /* 4read pathname */
        fgets(buff, MAXLINE, stdin);
        n = strlen(buff);               /* fgets() guarantees null byte at end */
        if (buff[n-1] == '\n')
                n--;                            /* delete newline from fgets() */

        snprintf(command, sizeof(command), "cat %s", buff);
        fp = popen(command, "r");

                /* 4copy from pipe to standard output */
        while (fgets(buff, MAXLINE, fp) != NULL)
                fputs(buff, stdout);

        pclose(fp);
        exit(0);
}

gcc -g -o mainpopen mainpopen.c

./mainpopen

/etc/issue

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值