细说linux IPC(六):pipe和FIFO

【版权声明:尊重原创,转载请保留出处:blog.csdn.net/shallnet 或 .../gentleliu,文章仅供学习交流,请勿用于商业用途】
在unix系统上最早的IPC形式为管道,管道的创建使用pipe函数:
      #include <unistd.h>
       int pipe(int pipefd[2]);

该函数创建一个单向的管道,返回两个描述符 pipefd[0],和pipefd[1],pipefd[0]用于读操作,pipefd[1]用于写操作。该函数一般应用在父子进程(有亲缘关系的进程)之间的通信,先是一个进程创建管道,再fork出一个子进程,然后父子进程可以通过管道来实现通信。
管道具有以下特点:
管道是半双工的,数据只能向一个方向流动;需要双方通信时,需要建立起两个管道;
只能用于父子进程或者兄弟进程之间(具有亲缘关系的进程);
单独构成一种独立的文件系统:管道对于管道两端的进程而言,就是一个文件,但它不是普通的文件,它不属于某种文件系统,而是自立门户,单独构成一种文件系统,并且只存在与内存中。
数据的读出和写入:一个进程向管道中写的内容被管道另一端的进程读出。写入的内容每次都添加在管道缓冲区的末尾,并且每次都是从缓冲区的头部读出数据。


函数pipe一般使用步骤如下:
1.pipe创建管道;
2.fork创建子进程;
3.父子进程分别关闭掉读和写(或写和读)描述符;
4.读端在读描述符上开始读(或阻塞在读上等待写端完成写),写端开始写,完成父子进程通信过程。
一个简单的通信实现(来自linux man手册的修改):

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


int main(int argc, char *argv[])
{
   int pipefd[2];
   pid_t cpid;
   char buf[128];
   int readlen;


   if (argc != 2) {
        fprintf(stderr, "Usage: %s <string>\n", argv[0]);
        return -1;
   }
   if (pipe(pipefd) < 0) {
       fprintf(stderr, "pipe: %s\n", strerror(errno));
       return -1;
   }
   cpid = fork();
   if (cpid < 0) {
       fprintf(stderr, "fork: %s\n", strerror(errno));
       return -1;
   }
   if (0 == cpid) { /* 子进程 */
       close(pipefd[1]); /* 子进程关闭写端 */
       readlen = read(pipefd[0], buf, 128); //子进程阻塞在读上,等待父进程写
       if (readlen < 0) {
            fprintf(stderr, "read: %s\n", strerror(errno));
            return -1;
       }
       write(STDOUT_FILENO, buf, readlen);
       write(STDOUT_FILENO, "\n", 1);
       close(pipefd[0]); //读完之后关闭读描述符
       return 0;
   } else { /* 父进程 */
       close(pipefd[0]); /*父进程关闭没用的读端 */
       sleep(2);
       write(pipefd[1], argv[1], strlen(argv[1])); //父进程开始写
       close(pipefd[1]); /* 父进程关闭写描述符 */
       wait(NULL); /* 父进程等待子进程退出,回收子进程资源 */
       return 0;
   }
}

运行时将打印命令行输入参数,打印将在父进程睡眠2秒之后,子进程将阻塞在读,直到父进程写完数据,可见管道是有同步机制的,不需要自己添加同步机制。如果希望两个进程双向数据传输,那么需要建立两个管道来实现。
管道最大的劣势就是只能在拥有共同祖先进程的进程之间通信,在无亲缘关系的两个进程之间没有办法使用,不过有名管道FIFO解决了这个问题。FIFO类似于pipe,也是只能单向传输数据,不过和pipe不同的是他可以在无亲缘关系的进程之间通信,它提供一个路径与之关联,所以只要能访问该路径的进程都可以建立起通信,类似于前面的共享内存,都提供一个路径与之关联。
       #include <sys/types.h>
       #include <sys/stat.h>
       int mkfifo(const char *pathname, mode_t mode);
pathname 为系统路径名,mode为文件权限位,类似open函数第二个参数。
打开或创建一个新的fifo是先调用mkfifo,当指定的pathname已存在fifo时,mkfifo返回EEXIST错误,此时再调用open函数。
下面来使用mkfifo实现一个无亲缘关系进程间的双向通信,此时需要建立两个fifo,分别用于读写。服务进程循环的读并等待客户进程写,之后打印客户进程传来数据并向客户进程返回数据;客户进程向服务器写数据并等待读取服务进程返回的数据。
server process:
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include "slnipc.h"
int main(int argc, const char *argv[])
{
    int rc;
    int wr_fd, rd_fd;
    char sendbuf[128];
    char recvbuf[128];
    rc = mkfifo(SLN_IPC_2SER_PATH, O_CREAT | O_EXCL); //建立服务进程读的fifo
    if ((rc < 0 ) && (errno != EEXIST)) {
        fprintf(stderr, "mkfifo: %s\n", strerror(errno));
        return -1;
    }
    rc = mkfifo(SLN_IPC_2CLT_PATH, O_CREAT | O_EXCL); //建立服务进程写的fifo
    if ((rc < 0 ) && (errno != EEXIST)) {
        fprintf(stderr, "mkfifo: %s\n", strerror(errno));
        return -1;
    }
    wr_fd = open(SLN_IPC_2CLT_PATH, O_RDWR, 0);
    if (wr_fd < 0) {
        fprintf(stderr, "open: %s\n", strerror(errno));
        return -1;
    }
    rd_fd = open(SLN_IPC_2SER_PATH, O_RDWR, 0);
    if (rd_fd < 0) {
        fprintf(stderr, "open: %s\n", strerror(errno));
        return -1;
    }
    for (;;) {
        rc = read(rd_fd, recvbuf, sizeof(recvbuf)); //循环等待接受客户进程数据
        if (rc < 0) {
            fprintf(stderr, "read: %s\n", strerror(errno));
            continue;
        }
        printf("server recv: %s\n", recvbuf);
        snprintf(sendbuf, sizeof(sendbuf), "Hello, this is server!\n");
        rc = write(wr_fd, sendbuf, strlen(sendbuf));
        if (rc < 0) {
            fprintf(stderr, "write: %s\n", strerror(errno));
            continue;
        }
    }
    close(wr_fd);
    close(rd_fd);
    return 0;
}

client process:

#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <string.h>
#include <errno.h>
#include "slnipc.h"
int main(int argc, const char *argv[])
{
    int rc;
    int rd_fd, wr_fd;
    char recvbuf[128];
    char sendbuf[128];
    if (argc != 2) {
        fprintf(stderr, "Usage: %s <string>\n", argv[0]);
        return -1;
    }
    snprintf(sendbuf, sizeof(sendbuf), "%s", argv[1]);
    wr_fd = open(SLN_IPC_2SER_PATH, O_RDWR, 0);
    if (wr_fd < 0) {
        fprintf(stderr, "open: %s\n", strerror(errno));
        return -1;
    }
    rd_fd = open(SLN_IPC_2CLT_PATH, O_RDWR, 0);
    if (rd_fd < 0) {
        fprintf(stderr, "open: %s\n", strerror(errno));
        return -1;
    }
    rc = write(wr_fd, sendbuf, strlen(sendbuf));
    if (rc < 0) {
        fprintf(stderr, "write: %s\n", strerror(errno));
        return -1;
    }
    rc = read(rd_fd, recvbuf, sizeof(recvbuf));
    if (rc < 0) {
        fprintf(stderr, "write: %s\n", strerror(errno));
        return -1;
    }
    printf("client read: %s\n", recvbuf);
    close(wr_fd);
    close(rd_fd);
    return 0;
}

服务器先启动运行,之后运行客户端,运行结果
# ./server 
server recv: hi,this is fifo client 


# ./client "hi,this is fifo client" 
client read: Hello, this is server! 


这里有一些类似于socket实现进程间通信过程,只是fifo的读写描述符是两个,socket的读写使用同一个描述符。fifo的出现克服了管道的只能在有亲缘关系的进程之间的通信。和其他的进程间通信一直,fifo传送的数据也是字节流,需要自己定义协议格式来解析通信的数据,可以使用socket章节介绍的方式来实现的通信协议。

本节源码下载:

http://download.csdn.net/detail/gentleliu/8183027


郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。