文章目录
- 一、前言
- 二、命名管道的使用
一、前言
我们上篇博客里谈到,进程间通信的本质在于如何让两个进程看到同一份资源。匿名管道的核心思想就是让子进程继承父进程,从而让父子进程看到同一份管道文件,但这也使通信仅仅局限在具有血缘关系的进程之间。
而命名管道的处理办法就比较直截了当了:在指定路径下创建FIFO管道文件。多个进程就像使用普通文件一样对同一份管道文件进行读写操作,由此实现进程间通信。虽然命名管道挂靠于文件系统,但是内核在传递数据的时候会和磁盘去关联,写入时向内存中写入,读取时也从内存中读取,这和匿名管道是一致的
二、命名管道的使用
[参数说明]:
- pathname:指定管道文件的路径
- mode:指定文件的读写权限。管道文件的实际权限 =
mode & umask
,系统默认掩码为0002
[返回值]: 创建成功返回0,失败返回-1
[使用总结]:
- 以
O_RDONLY O_WRONLY
方式打开文件
在进行任何输入和输出操作之前,命名管道的读写端必须都被打开,否则管道会被阻塞 - 非阻塞式的打开文件
O_NONBLOCK
以只读的方式打开会成功;以只写的方式打开会报ENXIO(no such device or address)的错误
[demo]:
1.Comm.hpp文件
#include <iostream>
#include <cstring>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#define FIFO_PATH "./FIFO"
using namespace std;
void CreateFifo()
{
if(access(FIFO_PATH, F_OK) != 0) // 管道文件不存在才需要创建
{
if(mkfifo(FIFO_PATH, 0666) != 0) // 创建管道文件失败
{
perror("mkfifo");
}
}
}
2.ServiceFifo文件
#include "Comm.hpp"
int main()
{
CreateFifo();
int fd = open(FIFO_PATH, O_RDONLY);
while(true)
{
char buff[15] = {0};
ssize_t ss = read(fd, buff, sizeof(buff));
buff[ss] = '\0';
cout << buff << endl;
}
}
3.ClientFIFO.cc文件
#include "Comm.hpp"
int main()
{
int fd = open(FIFO_PATH, O_WRONLY);
int cnt = 0;
while(true)
{
char msg[15] = "hello world";
write(fd, msg, sizeof(msg));
sleep(1);
}
return 0;
}