Linux--进程间通讯--FIFO(open打开)

1. 什么是FIFO

FIFO命名管道,也叫有名管道 ,来区分管道pipe 。管道pipe只能用于有血缘关系的进程间通信,但通过FIFO可以实现不相关的进程之间交换数据。FIFO是Linux基础文件类型中的一种,但是FIFO文件在磁盘上没有数据块,仅仅用来标识内核中的一条通道。各进程可以打开这个文件进行read/write操作,实际上是在读写内核通道,这样就实现了进程间通信。

创建FIFO的方式:

使用命令创建:mkfifo 管道名,可以理解为创建一个管道伪文件。

使用库函数创建:mkfifo()函数,并且一旦使用mkfifo()创建了一个FIFO,就可以使用open来打开它,常见的文件I/O函数都可用于FIFO。如:close、read、write、unlink等。

实际上,创建一个FIFO命名管道的时候,内核会为FIFO(伪)文件开辟一个缓冲区,操作FIFO文件就相当于操作这个缓冲区,以此来实现进程间的通信,这种通信实际上就是文件读写的操作来实现的。(可以把FIFO理解为一个文件,一个进程向该文件写数据,另一个进程从该文件中读书数据,前提是两个进程读写的是同一个FIFO文件才能实现通信)

2.当open一个FIFO时,是否设置非阻塞标志(O_NONBLOCK)的区别:

若没有执行O_NONBLOCK(默认),只读open要阻塞到某个其他进程为写而打开此FIFO。类似的,只写open要阻塞到其他进程为读而打开它。

若指定了O_NONBLOCK,则只读open立即返回。而只写open将出错返回-1,如果没有进程已经为读而打开该FIFO,其errno置ENXIO。

3.代码:

创建两个进程,一个进程向FIFO 数据,一个进程从FIFO数据。

write:

cpp 复制代码
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <unistd.h>
#include <string.h>
#include <fcntl.h>



//       int mkfifo(const char *pathname, mode_t mode);
//   ssize_t write(int fd, const void *buf, size_t count);



int main()
{

	int cnt = 0;
	char *str = "message from fifo";

	int fd = open("./ipc-file",O_WRONLY);
	printf("write open success\n");

	while(1){


		write(fd,str,strlen(str));
		sleep(2);
		cnt++;
		if(cnt == 5){
			break;



		}


	}

	close(fd);

	return 0;

}

read:

cpp 复制代码
#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <errno.h>
#include <fcntl.h>

//       int mkfifo(const char *pathname, mode_t mode);

int main()
{
	int cnt = 0;
	char buf[30]={0};	

	if(mkfifo("./ipc-file",0700)==-1 && errno != EEXIST){
		printf("mkfifo fail");
		perror("why");
	}	


	int fd = open("./ipc-file",O_RDONLY);
	printf("open success\n");


	while(1){

		int nread = read(fd,buf,30);
		printf("read %d byte from fifo/n,context :%s\n",nread,buf);
		cnt++;
		if(cnt == 3){

			break;


		}
	}

	close(fd);
	return 0;

}

结果:

相关推荐
库库的里昂17 分钟前
Linux系统Docker部署开源在线协作笔记Trilium Notes与远程访问详细教程
linux·运维·docker·开源
在下不上天28 分钟前
flume-将日志采集到hdfs
大数据·linux·运维·hadoop·hdfs·flume
mit6.8241 小时前
[Redis#3] 通用命令 | 数据类型 | 内部编码 | 单线程 | 快的原因
linux·redis·分布式
^Lim1 小时前
esp32 JTAG 串口 bootload升级
java·linux·网络
小林熬夜学编程2 小时前
【Linux系统编程】第五十弹---构建高效单例模式线程池、详解线程安全与可重入性、解析死锁与避免策略,以及STL与智能指针的线程安全性探究
linux·运维·服务器·c语言·c++·安全·单例模式
玖玖玖 柒染2 小时前
windows下sqlplus连接到linux oracle不成功
linux·windows·oracle
LuckyTHP2 小时前
CentOS 9 无法启动急救方法
linux·运维·centos
Bonne journée2 小时前
centos和ubuntu有什么区别?
linux·ubuntu·centos
vvw&2 小时前
如何在 Ubuntu 22.04 上安装带有 Nginx 的 ELK Stack
linux·运维·nginx·ubuntu·elk·elasticsearch·开源项目
Linux运维老纪2 小时前
交换机配置从IP(Switch Configuration from IP)
linux·服务器·网络·安全·运维开发·ip