linux命名管道fifo通信示例

目录

概述

要点

1 虽然fifo是一个文件,但是两个进程使用fifo时,各自都要用open函数打开fifo。

2 open有读写等多种模式。此外模式之间还可以组合。

读(O_RDONLY),不论是否阻塞

写+阻塞(O_WRONLY)

写+非阻塞(O_WRONLY | O_NONBLOCK)

3 程序启动次序

4 程序退出

代码

主进程:

子进程:


概述

管道适用于进程间通信。按照陈硕所著的《Linux多线程服务端编程:使用muduo C++网络库》3.4节的说法,“进程间最好使用TCP”。但是在实际工作中,由于各种原因,我还是喜欢使用管道或者其他方式完成进程间通信。

这里给出一个命名管道的通信示例。命名管道fifo是一个文件,在程序master(主进程)中创立,其名字由mkfifo函数定下来。主进程负责写入数据到管道。程序slave(子进程)负责读取数据,并显示。退出时,用户要在主进程中键入quit指令。两个进程各自使用close函数结束对fifo的访问。并且由读进程调用unlink函数删除fifo文件。

两个程序各自占据一个console。

要点

1 虽然fifo是一个文件,但是两个进程使用fifo时,各自都要用open函数打开fifo。

2 open有读写等多种模式。此外模式之间还可以组合。

读(O_RDONLY),不论是否阻塞

open处于读模式下,尝试打开一个不存在的fifo将导致open直接返回-1,打开失败。

写+阻塞(O_WRONLY)

open处于写模式+阻塞下,假如子进程还没有用读模式的open来打开fifo,则写模式的open将阻塞,直到fifo被子进程读模式的open打开。 

写+非阻塞(O_WRONLY | O_NONBLOCK)

open处于写模式+非阻塞下,假如子进程还没有用读模式的open来打开fifo,则写模式的open将直接返回-1,打开失败。

3 程序启动次序

在fifo没有创建时就让slave文件(子进程)open管道,尝试打开一个不存在的fifo将导致open直接返回-1,打开失败。下面的截图展示了这个情况:

linux命名管道fifo通信示例_第1张图片

所以应该先启动主进程来创建mkfifo。

open处于写模式+阻塞下试图打开fifo,假如子进程还没有用读模式的open来打开fifo,则写模式的open将阻塞,直到fifo被子进程读模式的open打开。下图展示了主进程(右)的阻塞。

linux命名管道fifo通信示例_第2张图片

只有在主进程先启动,随后启动子进程的情况下,两个进程的open才会先后成功打开fifo:

 

4 程序退出

 两个进程都要在退出前调用close函数关闭fifo文件。此外,读进程在close之后还要调用unlink删除fifo文件。

代码

主进程:

#include 
#include 
#include 
#include 
#include 
#include 
#include 

const char * pFifoName = "fifoM";

int main(void)
{
	if(access(pFifoName, F_OK) == -1)
	{
		//检查fifo文件是否存在,不存在就创建一个
		mkfifo(pFifoName, 0644);
	}

	int p = open(pFifoName, O_WRONLY);
	if(p != -1)
	{
		//假如读管道的进程还没就打开写模式的管道文件,则open将阻塞直到读进程启动
		std::cout<<"open"<

子进程:

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

const char * pFifoName = "fifoM";

int main(void)
{
	int p = open(pFifoName, O_RDONLY | O_NONBLOCK);

	if(p != -1)
	{
		char recv[128];
		while(true)
		{
			int iRet = read(p, recv, 127);
			if(-1 == iRet)
			{
				if(errno == EAGAIN)
				{
					//std::cout<<"again"< 0)
			{
				recv[iRet] = 0;
				std::cout<

你可能感兴趣的:(Linux,linux)