进程间通信系列 之 概述与对比   http://blog.csdn.net/younger_china/article/details/15808685

 进程间通信系列 之 共享内存及其实例   http://blog.csdn.net/younger_china/article/details/15961557

 进程间通信系列 之 共享内存简单实例   http://blog.csdn.net/younger_china/article/details/15991081

 进程间通信系列 之 信号(理论)   http://blog.csdn.net/younger_china/article/details/15976961

 进程间通信系列 之 信号实例   http://blog.csdn.net/younger_china/article/details/15968715

 进程间通信系列 之 信号综合实例   http://blog.csdn.net/younger_china/article/details/15980485

 进程间通信系列 之 命名管道FIFO及其应用实例   http://blog.csdn.net/younger_china/article/details/15808531

 进程间通信系列 之 管道(客户端和服务端通信)   http://blog.csdn.net/younger_china/article/details/15809281

 进程间通信系列 之 信号量详解及编程实例   http://blog.csdn.net/younger_china/article/details/15808531

 进程间通信系列 之 消息队列函数及其范例   http://blog.csdn.net/younger_china/article/details/15503871

 进程间通信系列 之 消息队列应用实例   http://blog.csdn.net/younger_china/article/details/15808501 

 进程间通信系列 之 socket套接字及其实例   http://blog.csdn.net/younger_china/article/details/15809163

 进程间通信系列 之 socket套接字实例   http://blog.csdn.net/younger_china/article/details/15809207

命名管道概述

如果我们要在不相关的进程间交换数据,那么使用FIFO文件将会十分方便。

FIFO文件通常也称为命名管道(named pipe)。命名管道是一种特殊类型的文件,它在文件系统中以文件名的形式存在。

创建命名管道

创建命名管道一般有两种方式:

命令行方式

一个比较旧的方式是:

mknod filename p

这个命令并未出现在X/Open规范的命令列表中,所以可能并不是所有的类Unix系统都可以这样做。

推荐的做法是:

mkfifo filename

函数调用方式

  1. #include <sys/types.h>
  2. #include <sys/stat.h>
  3. int mkfifo(const char *pathname, mode_t mode);
  4. int mknod(const char *pathname, mode_t mode | S_FIFO, (dev_t)0);

函数说明    mkfifo()会依参数pathname建立特殊的FIFO文件,该文件必须不存在,而参数mode为该文件的权限(mode%~umask),因此 umask值也会影响到FIFO文件的权限。Mkfifo()建立的FIFO文件其他进程都可以用读写一般文件的方式存取。当使用open()来打开 FIFO文件时,O_NONBLOCK旗标会有影响    1、当使用O_NONBLOCK 旗标时,打开FIFO 文件来读取的操作会立刻返回,但是若还没有其他进程打开FIFO 文件来读取,则写入的操作会返回ENXIO
错误代码。    2、没有使用O_NONBLOCK 旗标时,打开FIFO 来读取的操作会等到其他进程打开FIFO文件来写入才正常返回。同样地,打开FIFO文件来写入的操作会等到其他进程打开FIFO 文件来读取后才正常返回。返回值    若成功则返回0,否则返回-1,错误原因存于errno中。错误代码    EACCESS 参数pathname所指定的目录路径无可执行的权限    EEXIST 参数pathname所指定的文件已存在。    ENAMETOOLONG 参数pathname的路径名称太长。   
ENOENT 参数pathname包含的目录不存在    ENOSPC 文件系统的剩余空间不足    ENOTDIR 参数pathname路径中的目录存在但却非真正的目录。    EROFS 参数pathname指定的文件存在于只读文件系统内。

代码演示

下面代码演示了mkfifo函数的用法:

  1. #include <unistd.h>
  2. #include <stdlib.h>
  3. #include <stdio.h>
  4. #include <sys/types.h>
  5. #include <sys/stat.h>
  6. int main()
  7. {
  8. int res = mkfifo("~/Test/PipeTest/my_fifo",0777);
  9. if(!res)
  10. printf("FIFO created\n");
  11. exit(EXIT_SUCCESS);
  12. }

linux中ls命令的-F选项是列出文件的类型。

访问FIFO文件

同样有两种方式访问FIFO文件。

命令行方式

首先用cat命令读取刚才创建的FIFO文件:

cat < /tmp/my_fifo

这个时候,cat命令将一直挂起,直到终端或者有数据发送到FIFO中。

然后尝试向FIFO中写数据(在另外一个终端执行这个命令)

echo "FIFO test" > /tmp/my_fifo

这个时候cat将会输出内容。

函数调用方式

首先需要注意的是:

与通过pipe调用创建管道不同,FIFO是以命名文件的形式存在,而不是打开的文件描述符,所以在对它进行读写操作之前必须先打开它。

1、使用open函数打开FIFO文件

第一个是不能以O_RDWR模式打开FIFO文件进行读写操作。这样做的行为是未定义的。

因为我们通常使用FIFO只是为了单向传递数据,所以没有必要使用这个模式。

如果确实需要在程序之间双向传递数据,最好使用一对FIFO或管道,一个方向使用一个。或者采用先关闭在重新打开FIFO的方法来明确改变数据流的方向。

第二是对标志位的O_NONBLOCK选项的用法。

使用这个选项不仅改变open调用的处理方式,还会改变对这次open调用返回的文件描述符进行的读写请求的处理方式。

O_RDONLY、O_WRONLY和O_NONBLOCK标志共有四种合法的组合方式:

  • flags=O_RDONLY:open将会调用阻塞,除非有另外一个进程以写的方式打开同一个FIFO,否则一直等待。
  • flags=O_WRONLY:open将会调用阻塞,除非有另外一个进程以读的方式打开同一个FIFO,否则一直等待。
  • flags=O_RDONLY|O_NONBLOCK:如果此时没有其他进程以写的方式打开FIFO,此时open也会成功返回,此时FIFO被读打开,而不会返回错误。
  • flags=O_WRONLY|O_NONBLOCK:立即返回,如果此时没有其他进程以读的方式打开,open会失败打开,此时FIFO没有被打开,返回-1。

测试代码:

  1. <pre code_snippet_id="75537" snippet_file_name="blog_20131121_2_738143">

#include<unistd.h>#include<stdlib.h>#include<stdio.h>#include<string.h>#include<fcntl.h>#include<sys/types.h>#include<sys/stat.h>#define FIFO_NAME "/tmp/my_fifo"int main(int argc,char *argv[]){ int res,i; int open_mode=0; if(argc < 2){ fprintf(stderr,"Usage:%s<some
combination of \ O_RDONLY,O_WRONLY,O_NONBLOCK\n",*argv); exit(EXIT_FAILURE); } argv++; if(strncmp(*argv,"O_RDONLY",8)==0)open_mode|=O_RDONLY; if(strncmp(*argv,"O_WRONLY",8)==0)open_mode|=O_WRONLY; if(strncmp(*argv,"O_NONBLOCK",10)==0)open_mode|=O_NONBLOCK;
for(i = 1;i < argc;++i){ argv++; if(*argv){ if(strncmp(*argv,"O_RDONLY",8)==0)open_mode|=O_RDONLY; if(strncmp(*argv,"O_WRONLY",8)==0)open_mode|=O_WRONLY; if(strncmp(*argv,"O_NONBLOCK",10)==0)open_mode|=O_NONBLOCK; }} if(access(FIFO_NAME,F_OK)==-1){ res=mkfifo(FIFO_NAME,0777);
if(res!=0){ fprintf(stderr,"Could not create fifo %s\n",FIFO_NAME); exit(EXIT_FAILURE); } } printf("process %d open FIFO with %d\n",getpid(),open_mode); res=open(FIFO_NAME,open_mode); printf("process %d result %d\n",getpid(),res); sleep(5); if(res!=-1)close(res);
printf("process %d finished\n",getpid()); exit(EXIT_SUCCESS);}

对FIFO文件进行读写操作

open函数调用中的参数标志O_NONBLOCK会影响FIFO的读写操作。

规则如下:

  • 对一个空的阻塞的FIFO的read调用将等待,直到有数据可以读的时候才继续执行/
  • 对一个空的非阻塞的FIFO的read调用立即返回0字节。
  • 对一个完全阻塞的FIFO的write调用将等待,直到数据可以被写入时才开始执行。
    • 系统规定:如果写入的数据长度小于等于PIPE_BUF字节,那么或者写入全部字节,要么一个字节都不写入。

注意这个限制的作用:

当只使用一个FIF并允许多个不同的程序向一个FIFO读进程发送请求的时候,为了保证来自不同程序的数据块 不相互交错,即每个操作都原子化,这个限制就很重要了。如果能够包子所有的写请求是发往一个阻塞的FIFO的,并且每个写请求的数据长父小于等于PIPE_BUF字节,系统就可以确保数据绝不会交错在一起。通常将每次通过FIFO传递的数据长度限制为PIPE_BUF是一个好办法。

  • 在非阻塞的write调用情况下,如果FIFO 不能接收所有写入的数据,将按照下面的规则进行:

    • 请求写入的数据的长度小于PIPE_BUF字节,调用失败,数据不能被写入。
    • 请求写入的数据的长度大于PIPE_BUF字节,将写入部分数据,返回实际写入的字节数,返回值也可能是0。

其中。PIPE_BUF是FIFO的长度,它在头文件limits.h中被定义。在linux或其他类UNIX系统中,它的值通常是4096字节。

在这里需要注意的是:

有两个进程去访问FIFO管道时,Linux会安排好两个进程之间的调度,使得两个进程在可以运行的时候运行,在不能运行的时候阻塞。

程序实例

下面的程序用命名管道在两个独立的进程之间通信,模拟了消费者和生产者程序。

生产者程序fork3.c:

  1. //生产者程序
  2. #include <unistd.h>
  3. #include <stdlib.h>
  4. #include <stdio.h>
  5. #include <string.h>
  6. #include <fcntl.h>
  7. #include <limits.h>
  8. #include <sys/types.h>
  9. #include <sys/stat.h>
  10.  
  11. #define FIFO_NAME "/tmp/my_fifo"
  12. #define BUFFER_SIZE PIPE_BUF
  13. #define TEN_MSG (1024 * 1024 * 10)
  14.  
  15. int main()
  16. {
  17. int pipe_fd;
  18. int res;
  19. int open_mode = O_WRONLY;
  20. int bytes_sent = 0;
  21. char buffer[BUFFER_SIZE + 1];
  22.  
  23. printf("Productor Program beginning...\n");
  24.  
  25. //检查FIFO文件是否存在
  26. if(access(FIFO_NAME, F_OK) == -1){
  27. res = mkfifo(FIFO_NAME, 0777);
  28. if(res != 0){
  29. fprintf(stderr, "Could not create fifo %s\n", FIFO_NAME);
  30. exit(EXIT_FAILURE);
  31. }
  32. }
  33. printf("Process %d opening FIFO O_WRONLY\n", getpid());
  34. //打开FIFO文件
  35. pipe_fd = open(FIFO_NAME, open_mode);
  36. printf("Process %d result %d\n", getpid(), pipe_fd);
  37.  
  38. if(pipe_fd != -1){
  39. while(bytes_sent < TEN_MSG){
  40. res = write(pipe_fd, buffer, BUFFER_SIZE); //向FIFO写入数据
  41. if(res == -1){
  42. fprintf(stderr, "Write error on pipe\n");
  43. exit(EXIT_FAILURE);
  44. }
  45. bytes_sent += res;
  46. }
  47. (void)close(pipe_fd);
  48. }
  49. else{
  50. exit(EXIT_FAILURE);
  51. }
  52.  
  53. printf("Process %d finished\n", getpid());
  54. exit(EXIT_SUCCESS);
  55. }

消费者程序fork4.c:

  1. //消费者程序
  2.  
  3. #include <stdio.h>
  4. #include <stdlib.h>
  5. #include <string.h>
  6. #include <fcntl.h>
  7. #include <limits.h>
  8. #include <sys/types.h>
  9. #include <sys/stat.h>
  10.  
  11. #define FIFO_NAME "/tmp/my_fifo"
  12. #define BUFFER_SIZE PIPE_BUF
  13.  
  14. int main()
  15. {
  16. int pipe_fd;
  17. int res;
  18.  
  19. int open_mode = O_RDONLY;
  20. char buffer[BUFFER_SIZE + 1];
  21. int bytes = 0;
  22.  
  23. printf("COnsumer Program beginning...");
  24. memset(buffer,'\0', sizeof(buffer));
  25.  
  26. printf("Process %d opeining FIFO O_RDONLY\n", getpid());
  27. pipe_fd = open(FIFO_NAME, open_mode);
  28. printf("Process %d result %d\n", getpid(), pipe_fd);
  29.  
  30. if (pipe_fd != -1)
  31. {
  32. do{
  33. res = read(pipe_fd, buffer, BUFFER_SIZE);
  34. bytes += res;
  35. }while(res > 0);
  36. close(pipe_fd);
  37. }
  38. else
  39. {
  40. exit(EXIT_FAILURE);
  41. }
  42.  
  43. printf("Process %d finished, %d bytes read\n", getpid(), bytes);
  44. exit(EXIT_SUCCESS);
  45. }

运行结果:

可以发现读进程只运行了不到0.1S的时间,却读取了10MB的数据。这说明管道在程序之间传递数据是很有效率的。

删除FIFO文件

FIFO文件使用完毕之后需删除,以免造成垃圾文件。

unlink(2) - Linux man page

如需转载,请注明出处:http://blog.csdn.net/xiajun07061225/article/details/8471777


  1.  
  1.  
  1.  
  1.  

进程间通信系列 之 命名管道FIFO及其应用实例的更多相关文章

  1. 进程间通信系列 之 socket套接字及其实例

    进程间通信系列 之 概述与对比   http://blog.csdn.net/younger_china/article/details/15808685  进程间通信系列 之 共享内存及其实例   ...

  2. Linux进程间通信之管道(pipe)、命名管道(FIFO)与信号(Signal)

    整理自网络 Unix IPC包括:管道(pipe).命名管道(FIFO)与信号(Signal) 管道(pipe) 管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道 ...

  3. 进程间通信___命名管道(FIFO)

    命名管道(FIFO) 基本概念 命名管道和一般的管道基本相同,但也有一些显著的不同: 命名管道是在文件系统中作为一个特殊的设备文件而存在的. 不同祖先的进程之间可以通过管道共享数据. 当共享管道的进程 ...

  4. Linux学习笔记25——命名管道(FIFO)

    1 命名管道(FIFO) 管道应用的一个重大缺陷就是没有名字,因此只能用于亲缘进程之间的通信.后来从管道为基础提出命名管道(named pipe,FIFO)的概念,该限制得到了克服.FIFO不同于管道 ...

  5. 进程间通信:命名管道FIFO(2)

    一.命名管道 如果我们想在不相关的进程之间交换数据,可以用FIFO文件来完成这项工作,它通常也被称为命名管道.命名管道是一种特殊类型的文件,它在文件系统中以文件名的形式存在,但是它的行为却和我们已经见 ...

  6. 命名管道FIFO和mkfifo函数

    进程间通信必须通过内核提供的通道,而且必须有一种办法在进程中标识内核提供的某个通道,前面讲过的匿名管道是用打开的文件描述符来标识的.如果要互相通信的几个进程没有从公共祖先那里继承文件描述符,它们怎么通 ...

  7. 命名管道FIFO

    首先我得检讨一下自己,这几天有些颓呀,打不起精神,板子出了点问题,果真自学还是很困难呀,硬件方面难解决呀,理想与现实还是很有差距的,伤透了,凌乱了. 一直在理解进程间通信的问题.发现上次忽略了一个问题 ...

  8. 命名管道FIFO及其读写规则

    一.匿名管道的一个限制就是只能在具有共同祖先的进程间通信命名管道(FIFO):如果我们想在不相关的进程之间切换数据,可以使用FIFO文件来做这项工作注意:命名管道是一种特殊类型文件.利用命令:$ mk ...

  9. 进程间通信系列 之 socket套接字实例

    进程间通信系列 之 概述与对比   http://blog.csdn.net/younger_china/article/details/15808685  进程间通信系列 之 共享内存及其实例   ...

随机推荐

  1. Yii2.0修改默认控制器

    设置默认控制器有两种方法 1.在/vendor/yiisoft/yii2/web/Application.PHP的第28行左右 public $defaultRoute = 'site';      ...

  2. 使用 @Qualifier 注释和 @Autowired 注释通过指定哪一个真正的 bean 将会被装配来消除混乱

    1.当你创建多个具有相同类型的 bean 时,并且想要用一个属性只为它们其中的某一个进行装配,在这种情况下,你可以使用 @Qualifier 注释和 @Autowired 注释来精确配置. 2.示例 ...

  3. TCP/IP笔记(二)TCP/IP简介

    上回,主要介绍了下协议和OSI参考模型,并简单了解下网络构成要素,这回该说说TCP/IP了 互联网与TCP/IP的关系   互联网进行通信时,需要相应的网络协议,TCP/IP原本就是为使用互联网而开发 ...

  4. 深入浅出数据结构C语言版(4)——表与链表

    在我们谈论本文具体内容之前,我们首先要说明一些事情.在现实生活中我们所说的"表"往往是二维的,比如课程表,就有行和列,成绩表也是有行和列.但是在数据结构,或者说我们本文讨论的范围内 ...

  5. 解决华为手机不打印Log信息的问题

    在之前安装了Android Studio后,发现了一个很苦恼的事情,就是在程序中的写Log语句,不能正常的在Logcat中打印出来,这对于解决程序bug真是一刀切断,让人无从下手,在各种尝试后,首先我 ...

  6. iOSiOS开发之退出功能(易错)

    如果,我们有两个控制器,第一个控制器是MainController,它是与Main.storyboard相关联的.第二个控制器是myController.假设myController中有一个退出按钮, ...

  7. C—动态内存分配之malloc与realloc的区别

    在程序的执行期间分配内存时,内存区域中的这个空间称为堆(heap).还有另一个内存区域,称为堆栈(stack),其中的空间分配给函数的参数和本地变量.在执行完该函数后,存储参数和本地变量的内存空间就会 ...

  8. 解决mac os下mcss命令报错:env: node\r: No such file or directory

    标题无“转载”即原创文章,版权所有.转载请注明来源:http://besteam.im/blogs/article/31/. 我一直对字符界面有抵触感,即使会用vim,我的linux脚本(python ...

  9. linux之date命令详解

    date命令的用处 1.用于显示特殊的时间格式,可以用于对日志文件的命名 2.用于设置时间 ,不过这方面用的比较少,因为一般的服务器都设置的有自动同步网络时间 用法: date [OPTION]... ...

  10. Modbus通信协议的压力测试

    最近物联网都比较的火,因此,特别为各位兄弟姐妹们,奉上一款Mobus协议的测试软件,可以用来做设备的压力测试,和通信测试. 起初软件开发缘由是我们最近在开发一款设备,需要将多个DS18B20并联起来, ...