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

前言

我们在之前学习了匿名管道与匿名管道的应用——进程池,但是匿名管道的通信,需要有血缘关系的进程(通过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

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

 

 

 

 

 

 

 

 

相关推荐

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-03-20 23:24:05       98 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-03-20 23:24:05       106 阅读
  3. 在Django里面运行非项目文件

    2024-03-20 23:24:05       87 阅读
  4. Python语言-面向对象

    2024-03-20 23:24:05       96 阅读

热门阅读

  1. 小项目知识点

    2024-03-20 23:24:05       46 阅读
  2. AcWing 167.木棒

    2024-03-20 23:24:05       47 阅读
  3. 2024最新华为OD机试试题库全 -【游戏分组】- C卷

    2024-03-20 23:24:05       48 阅读
  4. MongoDB聚合运算符:$floor

    2024-03-20 23:24:05       45 阅读
  5. 安卓面试题多线程 61-65

    2024-03-20 23:24:05       35 阅读
  6. Typescript泛型

    2024-03-20 23:24:05       42 阅读
  7. 5.1.1.1、【AI技术新纪元:Spring AI解码】功能调用

    2024-03-20 23:24:05       37 阅读
  8. SpringBoot 如何快速过滤出一次请求的所有日志?

    2024-03-20 23:24:05       42 阅读
  9. rtt自动初始化机制学习

    2024-03-20 23:24:05       44 阅读
  10. Linux 系统编程

    2024-03-20 23:24:05       37 阅读