1、概述

  无名管道只能在具有亲缘关系的进程中使用,而有名管道可以在互不相关的两个进程间使用。有名管道将管道以文件的方式存储在指定路径中,使用ls -l可以看到第一个字符是‘p’,表示这是一个管道文件。文件操作用标准IO,即open,read,write,close。

2、函数介绍

2.1 创建管道文件

2.2.1 mkfifo

函数原型 int mkfifo(const char *pathname, mode_t mode);
头文件 sys/types.h、sys/stat.h
功能 创建一个管道文件
参数

[in]:pathname:文件路径

[in]:mode:文件权限

返回 成功返回0,失败返回-1

  与有名信号量不同,管道文件是可以自己指定路径的。文件的权限也是与掩码相关的,(mode & (~umask))才是真正的文件权限,umask的值可以在终端直接输入umask查看。使用mkfifo创建的文件必须是不存在的,否则会出错,并置errno为EEXIST,此外还有其他错误码,可以通过在终端输入man 3 mkfifo查看。

3、测试程序

fifo1.c

 1 /**
 2  * filename: fifo1.c
 3  * author: Suzkfly
 4  * date: 2021-02-10
 5  * platform: Ubuntu
 6  *     配合fifo2使用,在一个终端执行fifo1,输入数据,按回车结束,另开1个终端执行
 7  *     fifo2,从fifo1中输入的数据能从fifo2中读出来。
 8  */
 9 #include <stdio.h>
10 #include <fcntl.h>
11 #include <sys/types.h>
12 #include <sys/stat.h>
13 #include <string.h>
14 
15 int main(int argc, const char *argv[])
16 {
17     int ret = 0;
18     int fd = 0;
19     char buf[128] =  { 0 };
20     
21     /* 创建fifo文件 */
22 #if 1
23     ret = mkfifo("myfifo", 0666);
24     if (ret == -1) {
25         printf("mkfifo failed\n");
26         return 0;
27     }
28     printf("ret = %d\n", ret);
29 #endif
30     
31     /* 以只写方式打开fifo文件 */
32     fd = open("myfifo", O_WRONLY);
33     while (1) {
34         memset(buf, 0, sizeof(buf));
35         scanf("%s", buf);
36         ret = write(fd, buf, strlen(buf));
37         printf("write ret = %d\n", ret);
38     }
39     
40     close(fd);
41     
42     return 0;
43 }

fifo2.c

 1 /**
 2  * filename: fifo2.c
 3  * author: Suzkfly
 4  * date: 2021-02-10
 5  * platform: Ubuntu
 6  *     配合fifo1使用,在一个终端执行fifo1,输入数据,按回车结束,另开1个终端执行
 7  *     fifo2,从fifo1中输入的数据能从fifo2中读出来。
 8  */
 9 #include <stdio.h>
10 #include <fcntl.h>
11 #include <sys/types.h>
12 #include <sys/stat.h>
13 #include <string.h>
14 
15 int main(int argc, const char *argv[])
16 {
17     int ret = 0;
18     int fd = 0;
19     char buf[128] =  { 0 };
20     
21     /* 打开fifo文件,以只读方式打开 */
22     fd = open("myfifo", O_RDONLY);
23     while (1) {
24         memset(buf, 0, sizeof(buf));
25         ret = read(fd, buf, sizeof(buf));
26         printf("read ret = %d\n", ret);
27         printf("read data:%s\n", buf);  
28     }
29     
30     close(fd);
31     
32     return 0;
33 }

测试结果:

 代码分析:

  测试例程代码比较简单,fifo1用来写数据,fifo2用来读数据,执行程序时需要先执行fifo1再执行fifo2,在fifo1的终端中输入数据,以回车键结束,在fifo2的终端中可以打印数据。

4、注意事项

  1. 如果一个进程以只写方式打开管道文件,如果没有其他进程以带有的方式(只读/读写)打开管道文件,那么写数据的操作将会阻塞,直到管道文件以被带有读的方式打开。(写阻塞不是因为没有读数据,而是没有以带有读的方式打开管道文件)。但是如果有进程以带读的方式打开了管道文件,之后又将文件关闭了,此时写端再向管道写数据则会收到SIGPIPE(管道破裂)信号,默认情况下写数据的进程会直接退出。

  2. 如果一个进程以只读方式打开管道文件,如果没有其他进程以带有的方式(只写/读写)打开管道文件,并且往管道内写数据,那么读操作将会阻塞。但是如果有进程以带写的方式打开了管道文件,之后又将文件关闭了,此时读端再向管道读数据则会收到SIGPIPE(管道破裂)信号,默认情况下写数据的进程会直接退出。

  3. 如果一个进程以读写方式打开管道文件,那么即使没有其他进程以带有读的方式打开管道文件,它往管道写数据的操作也不会阻塞,并且可以自己将数据读出来。

  上面3条我总结一下,一个有名管道存在一个类似于建立连接的机制,有写无读时写阻塞,有读无写或未写时读阻塞,但一旦建立好连接,之后如果所有的写端关闭,那么读操作会导致管道破裂,如果所有的读端关闭,那么写操作会导致管道破裂。

 

  4. 同无名管道一样,有名管道的容量也是65536(64K)字节,建立连接后如果一次写操作将导致管道内的数据超过65536个字节,那么这次写操作将会阻塞。

 

  5. 如果创建的管道文件没有某种权限,那么以包含该种权限的的方式去打开它则会失败(比如管道文件只给了读权限,那么以只写方式或者读写方式打开都会失败)。