C/C++ 进程间通信 管道

使用匿名管道

一、什么是管道

如果你使用过Linux的命令,那么对于管道这个名词你一定不会感觉到陌生,因为我们通常通过符号“|"来使用管道,但是管理的真正定义是什么呢?管道是一个进程连接数据流到另一个进程的通道,它通常是用作把一个进程的输出通过管道连接到另一个进程的输入。

举个例子,在shell中输入命令:ls -l | grep string,我们知道ls命令(其实也是一个进程)会把当前目录中的文件都列出来,但是它不会直接输出,而是把本来要输出到屏幕上的数据通过管道输出到grep这个进程中,作为grep这个进程的输入,然后这个进程对输入的信息进行筛选,把存在string的信息的字符串(以行为单位)打印在屏幕上。

二、使用popen函数

1、popen函数和pclose函数介绍

有静就有动,有开就有关,与此相同,与popen函数相对应的函数是pclose函数,它们的原型如下:

  1. #include <stdio.h>
  2. FILE* popen (const char *command, const char *open_mode);
  3. int pclose(FILE *stream_to_close);

poen函数允许一个程序将另一个程序作为新进程来启动,并可以传递数据给它或者通过它接收数据。command是要运行的程序名和相应的参数。open_mode只能是"r(只读)"和"w(只写)"的其中之一。注意,popen函数的返回值是一个FILE类型的指针,而Linux把一切都视为文件,也就是说我们可以使用stdio I/O库中的文件处理函数来对其进行操作。

如果open_mode是"r",主调用程序就可以使用被调用程序的输出,通过函数返回的FILE指针,就可以能过stdio函数(如fread)来读取程序的输出;如果open_mode是"w",主调用程序就可以向被调用程序发送数据,即通过stdio函数(如fwrite)向被调用程序写数据,而被调用程序就可以在自己的标准输入中读取这些数据。

pclose函数用于关闭由popen创建出的关联文件流。pclose只在popen启动的进程结束后才返回,如果调用pclose时被调用进程仍在运行,pclose调用将等待该进程结束。它返回关闭的文件流所在进程的退出码。

2、例子

很多时候,我们根本就不知道输出数据的长度,为了避免定义一个非常大的数组作为缓冲区,我们可以以块的方式来发送数据,一次读取一个块的数据并发送一个块的数据,直到把所有的数据都发送完。下面的例子就是采用这种方式的数据读取和发送方式。源文件名为popen.c,代码如下:

  1. #include <unistd.h>
  2. #include <stdlib.h>
  3. #include <stdio.h>
  4. #include <string.h>
  5. int main()
  6. {
  7. FILE *read_fp = NULL;
  8. FILE *write_fp = NULL;
  9. char buffer[BUFSIZ + 1];
  10. int chars_read = 0;
  11. //初始化缓冲区
  12. memset(buffer, ‘\0‘, sizeof(buffer));
  13. //打开ls和grep进程
  14. read_fp = popen("ls -l", "r");
  15. write_fp = popen("grep rwxrwxr-x", "w");
  16. //两个进程都打开成功
  17. if(read_fp && write_fp)
  18. {
  19. //读取一个数据块
  20. chars_read = fread(buffer, sizeof(char), BUFSIZ, read_fp);
  21. while(chars_read > 0)
  22. {
  23. buffer[chars_read] = ‘\0‘;
  24. //把数据写入grep进程
  25. fwrite(buffer, sizeof(char), chars_read, write_fp);
  26. //还有数据可读,循环读取数据,直到读完所有数据
  27. chars_read = fread(buffer, sizeof(char), BUFSIZ, read_fp);
  28. }
  29. //关闭文件流
  30. pclose(read_fp);
  31. pclose(write_fp);
  32. exit(EXIT_SUCCESS);
  33. }
  34. exit(EXIT_FAILURE);
  35. }

3、popen的实现方式及优缺点

当请求popen调用运行一个程序时,它首先启动shell,即系统中的sh命令,然后将command字符串作为一个参数传递给它。

这样就带来了一个优点和一个缺点。优点是:在Linux中所有的参数扩展都是由shell来完成的。所以在启动程序(command中的命令程序)之前先启动shell来分析命令字符串,也就可以使各种shell扩展(如通配符)在程序启动之前就全部完成,这样我们就可以通过popen启动非常复杂的shell命令。

而它的缺点就是:对于每个popen调用,不仅要启动一个被请求的程序,还要启动一个shell,即每一个popen调用将启动两个进程,从效率和资源的角度看,popen函数的调用比正常方式要慢一些。

三、pipe调用

如果说popen是一个高级的函数,pipe则是一个底层的调用。与popen函数不同的是,它在两个进程之间传递数据不需要启动一个shell来解释请求命令,同时它还提供对读写数据的更多的控制。

pipe函数的原型如下:

  1. #include <unistd.h>
  2. int pipe(int file_descriptor[2]);

我们可以看到pipe函数的定义非常特别,该函数在数组中墙上两个新的文件描述符后返回0,如果返回返回-1,并设置errno来说明失败原因。

数组中的两个文件描述符以一种特殊的方式连接起来,数据基于先进先出的原则,写到file_descriptor[1]的所有数据都可以从file_descriptor[0]读回来。由于数据基于先进先出的原则,所以读取的数据和写入的数据是一致的。

特别提醒:

1、从函数的原型我们可以看到,它跟popen函数的一个重大区别是,popen函数是基于文件流(FILE)工作的,而pipe是基于文件描述符工作的,所以在使用pipe后,数据必须要用底层的read和write调用来读取和发送。

2、不要用file_descriptor[0]写数据,也不要用file_descriptor[1]读数据,其行为未定义的,但在有些系统上可能会返回-1表示调用失败。数据只能从file_descriptor[0]中读取,数据也只能写入到file_descriptor[1],不能倒过来。

例子:

首先,我们在原先的进程中创建一个管道,然后再调用fork创建一个新的进程,最后通过管道在两个进程之间传递数据。源文件名为pipe.c,代码如下:

  1. #include <unistd.h>
  2. #include <stdlib.h>
  3. #include <stdio.h>
  4. #include <string.h>
  5. int main()
  6. {
  7. int data_processed = 0;
  8. int filedes[2];
  9. const char data[] = "Hello pipe!";
  10. char buffer[BUFSIZ + 1];
  11. pid_t pid;
  12. //清空缓冲区
  13. memset(buffer, ‘\0‘, sizeof(buffer));
  14. if(pipe(filedes) == 0)
  15. {
  16. //创建管道成功
  17. //通过调用fork创建子进程
  18. pid = fork();
  19. if(pid == -1)
  20. {
  21. fprintf(stderr, "Fork failure");
  22. exit(EXIT_FAILURE);
  23. }
  24. if(pid == 0)
  25. {
  26. //子进程中
  27. //读取数据
  28. data_processed = read(filedes[0], buffer, BUFSIZ);
  29. printf("Read %d bytes: %s\n", data_processed, buffer);
  30. exit(EXIT_SUCCESS);
  31. }
  32. else
  33. {
  34. //父进程中
  35. //写数据
  36. data_processed = write(filedes[1], data, strlen(data));
  37. printf("Wrote %d bytes: %s\n", data_processed, data);
  38. //休眠2秒,主要是为了等子进程先结束,这样做也只是纯粹为了输出好看而已
  39. //父进程其实没有必要等等子进程结束
  40. sleep(2);
  41. exit(EXIT_SUCCESS);
  42. }
  43. }
  44. exit(EXIT_FAILURE);
  45. }

使用匿名管道,则通信的进程之间需要一个父子关系,通信的两个进程一定是由一个共同的祖先进程启动。但是匿名管道没有上面说到的数据交叉的问题。

与使用匿名管道相比,我们可以看到fifowrite.exe和fiforead.exe这两个进程是没有什么必然的联系的

使用命名管道

一、什么是命名管道

命名管道也被称为FIFO文件,它是一种特殊类型的文件,它在文件系统中以文件名的形式存在,但是它的行为却和之前所讲的没有名字的管道(匿名管道)类似。

由于Linux中所有的事物都可被视为文件,所以对命名管道的使用也就变得与文件操作非常的统一,也使它的使用非常方便,同时我们也可以像平常的文件名一样在命令中使用。

二、创建命名管道

我们可以使用两下函数之一来创建一个命名管道,他们的原型如下:

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

这两个函数都能创建一个FIFO文件,注意是创建一个真实存在于文件系统中的文件,filename指定了文件名,而mode则指定了文件的读写权限。

mknod是比较老的函数,而使用mkfifo函数更加简单和规范,所以建议在可能的情况下,尽量使用mkfifo而不是mknod。

三、访问命名管道

1、打开FIFO文件

与打开其他文件一样,FIFO文件也可以使用open调用来打开。注意,mkfifo函数只是创建一个FIFO文件,要使用命名管道还是将其打开。

但是有两点要注意,1、就是程序不能以O_RDWR模式打开FIFO文件进行读写操作,而其行为也未明确定义,因为如一个管道以读/写方式打开,进程就会读回自己的输出,同时我们通常使用FIFO只是为了单向的数据传递。2、就是传递给open调用的是FIFO的路径名,而不是正常的文件。

打开FIFO文件通常有四种方式,

  1. open(const char *path, O_RDONLY);//1
  2. open(const char *path, O_RDONLY | O_NONBLOCK);//2
  3. open(const char *path, O_WRONLY);//3
  4. open(const char *path, O_WRONLY | O_NONBLOCK);//4

在open函数的调用的第二个参数中,你看到一个陌生的选项O_NONBLOCK,选项O_NONBLOCK表示非阻塞,加上这个选项后,表示open调用是非阻塞的,如果没有这个选项,则表示open调用是阻塞的。

open调用的阻塞是什么一回事呢?很简单,对于以只读方式(O_RDONLY)打开的FIFO文件,如果open调用是阻塞的(即第二个参数为O_RDONLY),除非有一个进程以写方式打开同一个FIFO,否则它不会返回;如果open调用是非阻塞的的(即第二个参数为O_RDONLY | O_NONBLOCK),则即使没有其他进程以写方式打开同一个FIFO文件,open调用将成功并立即返回。

对于以只写方式(O_WRONLY)打开的FIFO文件,如果open调用是阻塞的(即第二个参数为O_WRONLY),open调用将被阻塞,直到有一个进程以只读方式打开同一个FIFO文件为止;如果open调用是非阻塞的(即第二个参数为O_WRONLY | O_NONBLOCK),open总会立即返回,但如果没有其他进程以只读方式打开同一个FIFO文件,open调用将返回-1,并且FIFO也不会被打开。

四、使用FIFO实现进程间的通信

说了这么多,下面就用一个例子程序来说明一下,两个进程如何通过FIFO实现通信吧。这里有两个源文件,一个fifowrite.c,它在需要时创建管道,然后向管道写入数据,数据由文件Data.txt提供,大小为10M,内容全是字符‘0’。另一个源文件为fiforead.c,它从FIFO中读取数据,并把读到的数据保存到另一个文件DataFormFIFO.txt中。为了让程序更加简洁,忽略了有些函数调用是否成功的检查。

fifowrite.c的源代码如下:

  1. #include <unistd.h>
  2. #include <stdlib.h>
  3. #include <fcntl.h>
  4. #include <limits.h>
  5. #include <sys/types.h>
  6. #include <sys/stat.h>
  7. #include <stdio.h>
  8. #include <string.h>
  9. int main()
  10. {
  11. const char *fifo_name = "/tmp/my_fifo";
  12. int pipe_fd = -1;
  13. int data_fd = -1;
  14. int res = 0;
  15. const int open_mode = O_WRONLY;
  16. int bytes_sent = 0;
  17. char buffer[PIPE_BUF + 1];
  18. if(access(fifo_name, F_OK) == -1)
  19. {
  20. //管道文件不存在
  21. //创建命名管道
  22. res = mkfifo(fifo_name, 0777);
  23. if(res != 0)
  24. {
  25. fprintf(stderr, "Could not create fifo %s\n", fifo_name);
  26. exit(EXIT_FAILURE);
  27. }
  28. }
  29. printf("Process %d opening FIFO O_WRONLY\n", getpid());
  30. //以只写阻塞方式打开FIFO文件,以只读方式打开数据文件
  31. pipe_fd = open(fifo_name, open_mode);
  32. data_fd = open("Data.txt", O_RDONLY);
  33. printf("Process %d result %d\n", getpid(), pipe_fd);
  34. if(pipe_fd != -1)
  35. {
  36. int bytes_read = 0;
  37. //向数据文件读取数据
  38. bytes_read = read(data_fd, buffer, PIPE_BUF);
  39. buffer[bytes_read] = ‘\0‘;
  40. while(bytes_read > 0)
  41. {
  42. //向FIFO文件写数据
  43. res = write(pipe_fd, buffer, bytes_read);
  44. if(res == -1)
  45. {
  46. fprintf(stderr, "Write error on pipe\n");
  47. exit(EXIT_FAILURE);
  48. }
  49. //累加写的字节数,并继续读取数据
  50. bytes_sent += res;
  51. bytes_read = read(data_fd, buffer, PIPE_BUF);
  52. buffer[bytes_read] = ‘\0‘;
  53. }
  54. close(pipe_fd);
  55. close(data_fd);
  56. }
  57. else
  58. exit(EXIT_FAILURE);
  59. printf("Process %d finished\n", getpid());
  60. exit(EXIT_SUCCESS);
  61. }

源文件fiforead.c的代码如下:

  1. #include <unistd.h>
  2. #include <stdlib.h>
  3. #include <stdio.h>
  4. #include <fcntl.h>
  5. #include <sys/types.h>
  6. #include <sys/stat.h>
  7. #include <limits.h>
  8. #include <string.h>
  9. int main()
  10. {
  11. const char *fifo_name = "/tmp/my_fifo";
  12. int pipe_fd = -1;
  13. int data_fd = -1;
  14. int res = 0;
  15. int open_mode = O_RDONLY;
  16. char buffer[PIPE_BUF + 1];
  17. int bytes_read = 0;
  18. int bytes_write = 0;
  19. //清空缓冲数组
  20. memset(buffer, ‘\0‘, sizeof(buffer));
  21. printf("Process %d opening FIFO O_RDONLY\n", getpid());
  22. //以只读阻塞方式打开管道文件,注意与fifowrite.c文件中的FIFO同名
  23. pipe_fd = open(fifo_name, open_mode);
  24. //以只写方式创建保存数据的文件
  25. data_fd = open("DataFormFIFO.txt", O_WRONLY|O_CREAT, 0644);
  26. printf("Process %d result %d\n",getpid(), pipe_fd);
  27. if(pipe_fd != -1)
  28. {
  29. do
  30. {
  31. //读取FIFO中的数据,并把它保存在文件DataFormFIFO.txt文件中
  32. res = read(pipe_fd, buffer, PIPE_BUF);
  33. bytes_write = write(data_fd, buffer, res);
  34. bytes_read += res;
  35. }while(res > 0);
  36. close(pipe_fd);
  37. close(data_fd);
  38. }
  39. else
  40. exit(EXIT_FAILURE);
  41. printf("Process %d finished, %d bytes read\n", getpid(), bytes_read);
  42. exit(EXIT_SUCCESS);
  43. }

但是为了数据的安全,我们很多时候要采用阻塞的FIFO,让写操作变成原子操作。

时间: 2024-08-28 20:19:34

C/C++ 进程间通信 管道的相关文章

Linux程序设计学习笔记----进程间通信——管道

转载请注明出处: http://blog.csdn.net/suool/article/details/38444149, 谢谢! 进程通信概述 在Linux系统中,进程是一个独立的资源管理单元,但是独立而不孤立,他们需要之间的通信,因此便需要一个进程间数据传递.异步.同步的机制,这个机制显然需要由OS来完成管理和维护.如下: 1.同一主机进程间数据交互机制:无名管道(PIPE),有名管道(FIFO),消息队列(Message Queue)和共享内存(Share Memory).无名管道多用于亲

Linux 进程间通信-管道

进程间通讯———管道 Linux 进程间通信-管道 进程是一个独立的资源分配单位,不同进程之间的资源是相互独立的,没有关联,不能在一个进程中直接访问另一个进程中的资源.但是,进程不是孤立的,不同的进程之间需要信息的交换以及状态的传递,因此需要进程间数据传递.同步与异步的机制. 此篇博文记录管道. 管道pipe 管道是进程间通信的主要手段之一.一个管道实际上就是个只存在于内存中的文件,对这个文件的操作要通过两个已经打开文件进行,它们分别代表管道的两端.管道是一种特殊的文件,它不属于某一种文件系统,

进程间通信---管道

每个进程各自有着不同的用户地址空间,任何一个进程的全局变量在另一个进程中是看不到的,所以进程之间要交换数据必须通过内核,在内核中开辟一块缓冲区,进程1把数据从用户空间拷到内核缓冲区,进程2在从内核中把数据读走,内核提供的这种机制称为进程间通信. 进程间通信的本质:让不同的进程看到同一份系统资源. 进程通信方式:管道(pipe) pipe函数: #include <unistd.h> int pipe(int fd[2]); 调用pipe函数时在内核中开辟一块缓冲区(管道)用于通信,它有一个读端

进程间通信 ---- 管道与FIFO 用法技巧

1.管道的创建 1.1 mkfifo(const char *pathname,mode_t mode); 函数已隐含指定O_CREAT|O_EXCL,所以它要么创建一个新的FIFO,要么返回EEXIST错误(已存在). 所以在创建已存在FIFO或新的FIFO,应该先调用mkfifo,并检查返回值 是否是EEXIST错误,若是EEXIST错误,则调      用open函数. 2.FIFO或管道读写 2.1对管道或FIFO的write 总是往末尾添加数据,对管道或FIFO的read总是从头开始读

Linux学习笔记(12)-进程间通信|管道

Linux的进程间通信有几种方式,包括,管道,信号,信号灯,共享内存,消息队列和套接字等-- 现在一个个的开始学习! -------------------------------------------------- 管道是一个进程链接另一个进程的数据通道,它通常是把一个进程的输出,接到另一个进程的输入,从而传递数据. 在Linux的终端上,用单竖线|来表示,那么,这个符号可以做什么呢? 举个栗子,如果我用ps -ef命令,可以查看我当前所有的进程: 正如上图表示,显示出来的东西太多了,让人眼

进程间通信——管道

1.管道 管道通信--只能在一台电脑上面运行. 管道:一定是半双工的通信,只能流向一个方向(规定流向): 管道是一个进程间通信的概念,在要通信的进程间构建一个单向的数据流动的通道.数据通过该通道从一个进程流向另一个进程时是具有时间先后顺序的.就像是在进程间架起了一个"管道". 模型分析 管道的实现:在Linux(Posix标准)的操作系统下,管道是通过文件来实现的.在后来的操作系统的发展中,依然使用了文件的访问方式来使用管道,但是具体的管道已经从外存挪到了内存. 2.无名管道   只能

进程间通信管道的应用

管道的概念: 管道是一种最基本的IPC机制,由pipe函数创建: #include<unistd.h> int pipe(int fileds[2]); 调用pipe函数时在内核中开辟一块缓冲区用于通信,它有一个读端一个写端,然后通过filedes参数传出给用户程序两个文件描述符,filedes[0]指向管道的读端,filedes[1]指向管道的写端. 进程在管道间通信: 1.父进程调用pipe开辟管道,得到两个文件描述符指向管道的两端. 2.父进程调用fork创建子进程,子进程有两个文件描述

【linux高级程序设计】(第九章)进程间通信-管道 3

有名管道 无名管道和有名管道: 1. 管道是特殊类型的文件,在满足先入先出的原则写可以读写,不能定位读写位置. 2.管道是单向的. 3.无名管道阻塞于读写位置,而有名管道阻塞在创建位置. 4.无名管道一般只用于亲缘关系进程间通信:有名管道以磁盘文件的方式存在,可以实现本机任意两进程间通信. shell创建有名管道 mknod 管道名 p  //创建名为PIPETEST的有名管道 mknod为命令 p是参数,表示有名管道 指令 > 管道名 &   //将指令结果输入到到管道文件中 指令 <

Linux进程间通信--管道

管道概述及相关API应用 1.1 管道相关的关键概念 管道是Linux支持的最初Unix IPC形式之一,具有以下特点: 管道是半双工的,数据只能向一个方向流动:需要双方通信时,需要建立起两个管道: 只能用于父子进程或者兄弟进程之间(具有亲缘关系的进程): 单独构成一种独立的文件系统:管道对于管道两端的进程而言,就是一个文件,但它不是普通的文件,它不属于某种文件系统,而是自立门户,单独构成一种文件系统,并且只存在与内存中. 数据的读出和写入:一个进程向管道中写的内容被管道另一端的进程读出.写入的

linux学习:进程间通信—管道

1.进程间通信当中一种比較简单的方法是管道操作 /* ============================================================================ Name : Test.c Author : wangchuan Version : Copyright : Your copyright notice Description : Hello World in C, Ansi-style ==========================