mkfifo函数的作用是在文件系统中创建一个文件,该文件用于提供FIFO功能,即命名管道。前边讲的那些管道都没有名字,因此它们被称为匿名管道,或简称管道。对文件系统来说,匿名管道是不可见的,它的作用仅限于在父进程和子进程两个进程间进行通信。而命名管道是一个可见的文件,因此,它可以用于任何两个进程之间的通信,不管这两个进程是不是父子进程,也不管这两个进程之间有没有关系。
1、创建有名管道
Mkfifo函数的原型如下所示:
#include <sys/types.h>
#include <sys/stat.h>
int mkfifo( const char *pathname, mode_t mode );
mkfifo函数需要两个参数,第一个参数(pathname)是将要在文件系统中创建的一个专用文件。第二个参数(mode)用来规定FIFO的读写权限。mkfifo函数如果调用成功的话,返回值为0;如果调用失败返回值为-1。
下面我们以一个实例来说明如何使用mkfifo函数建一个fifo,具体代码如下所示:
int ret;
... ret = mkfifo( "/tmp/cmd_pipe", S_IFIFO | 0666 );
if (ret == 0)
...{ // 成功建立命名管道 }
else
...{ // 创建命名管道失败 }
在这个例子中,利用/tmp目录中的cmd_pipe文件建立了一个命名管道(即fifo)。之后,就可以打开这个文件进行读写操作,并以此进行通信了。命名管道一旦打开,就可以利用典型的输入输出函数从中读取内容。举例来说,下面的代码段向我们展示了如何通过fgets函数来从管道中读取内容:
pfp = fopen( "/tmp/cmd_pipe", "r" );
... ret = fgets( buffer, MAX_LINE, pfp );
我们还能向管道中写入内容,下面的代码段向我们展示了利用fprintf函数向管道写入的具体方法:
pfp = fopen( "/tmp/cmd_pipe", "w+ );
... ret = fprintf( pfp, "Here’s a test string!/n" );
2、读写有名管道
#include <unistd.h>
ssize_t read (int fd , void * buf , size_t nbytes)
ssize_t write (int fd , void * buf , size_t nbytes)
在读写有名管道之前,需要用open函数打开该有名管道,打开有名管道操作与其他文件有一定的区别:如果希望打开管道的写端,则需要另一个进程打开管道的读端,如果只打开有名管道的一端,则系统将暂时阻塞打开进程,直到有另一个进程打开管道的另一端,当前进程才会继续执行,因此,在使用有名管道时,一定要使用两个进程分别打开其读端和写端。
非亲缘关系进程使用有名管道通信应用实例:
向有名管道中发送数据的进程代码:
从有名管道中接收数据的进程代码: