Linux——进程通信(三)命名管道

本文介绍了命名管道的原理,如何通过mkfifo创建管道,以及如何让两个没有直接关联的进程通过文件名进行通信。给出了服务器端和客户端的代码示例,展示了如何在C++中使用命名管道进行数据交换。
摘要由CSDN通过智能技术生成

前言

我们在之前学习了匿名管道与匿名管道的应用——进程池,但是匿名管道的通信,需要有血缘关系的进程(通过fork创建的进程们),如果我想让两个毫不相干的进程进行通信,可以采样命名管道的方式(有名字,通过名字可以找到,不需要继承的方式)。

一、命名管道的原理与指令

命名管道指令创建很简单,输入指令 mkfifo + 文件名,就可以创建一个命名管道。命名管道需要被创建在磁盘上,他是有自己的路径的,因为路径是具有唯一性的,所以我们可以使用路径+文件名的方式,来让不同进程看到同一份资源

那么两个没有血缘关系的进程,就可以通过命名管道的唯一性找到他,并借此通信。原理是跟匿名管道没什么区别,匿名管道是继承下来文件,命名管道是自己打开的文件

如下,就实现了命名管道的通信,echo进程与cat进程实现了通信。

同时,在我们管道通信的过程中,fifo管道的大小一直都是0,并不会写入到文件中,这是管道的特性决定了,因为根本没必要将数据拷贝到磁盘中,在缓冲区中就可以了,你写入就写在缓冲区中,你读取也在缓冲区中读取,这是内存级别的。

二、命名管道的代码

创建命名管道,首先需要mkfifo函数,第一个参数为创建命名管道的路径与文件名,第二个参数mode为创建的权限。创建成功返回0,失败返回-1,错误码也被设置。

代码思路:现在我们需要两个进程,不再是fork出来的,因此需要写一个服务端创建管道,并打开文件,读取客户端消息,还需要一个客户端打开服务端创建的管道文件,往里面写入消息。

具体代码调用接口就好。因为是复用的文件相关函数,因此代码实现没什么难度。

server.cc

#include <iostream>
#include <cstring>
#include <cerrno>
#include <cstdlib>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

using namespace std;

#define FILENAME "fifo"

int main()
{
    //创建命名管道
    int n = mkfifo(FILENAME,0666); 
    if(n<0)
    {
        cerr<<"错误码:"<<errno<<",错误原因:"<<strerror(errno)<<endl;
        return 1;
    }
    //打开文件
    int fd = open(FILENAME,O_RDONLY);
    if(fd < 0)
    {
        cerr<<"错误码:"<<errno<<",错误原因:"<<strerror(errno)<<endl;
        return 2;
    }
    //读取信息
    char buff[1024];
    while(true)
    {
        ssize_t rn = read(fd,buff,sizeof(buff)-1);
        if(rn > 0)
        {
            buff[rn] = '\0';
            cout<<"客户端说:"<<buff<<endl;
        }
    } 

}

client.cc 

#include <iostream>
#include <cstring>
#include <cerrno>
#include <cstdlib>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

using namespace std;
#define FILENAME "fifo"
int main()
{
    //打开文件
    int fd = open(FILENAME,O_WRONLY);
    if(fd<0)
    {
        cerr<<"错误码:"<<errno<<",错误原因:"<<strerror(errno)<<endl;
        return 1;
    }

    //写入消息
    string message;
    while(true)
    {
        getline(cin,message);
        ssize_t wn = write(fd,message.c_str(),message.size());
        if(wn<0)
        {
            cerr<<"错误码:"<<errno<<",错误原因:"<<strerror(errno)<<endl;
            break;
        }
    }

    close(fd);
    return 0;
}   

Makefile,由于我们需要两个进程共同协作,因此采样依赖的方式,先依赖上要形成的可执行文件,这样我们就可以一次生成两个可执行程序了。

.PHONY:all
all:server client

server:server.cc
	g++ -o $@ $^ -std=c++11

client:client.cc
	g++ -o $@ $^ -std=c++11

.PHONY:clean
clean:
	rm -f server client

执行这两个可执行程序,成功进行进程通信。

 

 

 

 

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值