前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >【Linux】进程间通信:命名管道

【Linux】进程间通信:命名管道

作者头像
用户11029103
发布2025-03-14 09:30:05
发布2025-03-14 09:30:05
4100
代码可运行
举报
文章被收录于专栏:技术学习技术学习
运行总次数:0
代码可运行

命名管道

管道应用的一个限制就是只能在具有共同祖先(具有亲缘关系)的进程间通信。 如果我们想在不相关的进程之间交换数据,可以使用FIFO文件来做这项工作,它经常被称为命名管道

命名管道的本质是一个特殊类型的文件,但它不会存储数据,只在内存中维护一个缓冲区,进行先进先出(FIFO)的数据传输

使用 mkfifo() 或 mknod() 在文件系统中创建一个特殊文件。

  • 一个进程以写的方式打开 FIFO,并向其中写入数据。
  • 另一个进程以读的方式打开 FIFO,并从中读取数据。
  • 数据是单向流动的

命名管道可以从命令行上创建,命令行方法是使用下面这个命令

代码语言:javascript
代码运行次数:0
运行
复制
mkfifo filename

p开头为管道文件

这里的读和写都是阻塞的,管道文件大小一直都是0,因为命名管道数据存储在内存中

命名管道也可以从程序里创建,相关函数:

代码语言:javascript
代码运行次数:0
运行
复制
int mkfifo(const char *filename,mode_t mode);

代码实现:首先完成管道生命周期的管理:

namedPipe.hpp文件,提供管道创建和关闭的函数:

代码语言:javascript
代码运行次数:0
运行
复制
#pragma once
#include<iostream>
#include <sys/types.h>
#include <sys/stat.h>
#include<string>
#include<cerrno>
#include<cstdio>
#include<unistd.h>
using namespace std;
const string comm_path ="./myfifo";

int CreateNamedPipe(const string & path)
{
    int res =mkfifo(path.c_str(),0666);
    if(res!=0)
    {
        perror("mkfifo");
    }
    return res;
}
int RemoveNamedPipe(const string & path)
{
    int res =unlink(path.c_str());
    if(res!=0)
    {
        perror("unlink");
    }
    return res;
}

server.cc:

代码语言:javascript
代码运行次数:0
运行
复制
#include"namedPipe.hpp"

int main()
{
    CreateNamedPipe(comm_path);
    RemoveNamedPipe(comm_path);
    return 0;
}

我们这里创建了两个文件,server.cc和client.cc,一定有一个文件来完成管道的创建和删除,这里生成可执行程序就是两个毫无关系的进程

这里可以对上面方法进行封装:

代码语言:javascript
代码运行次数:0
运行
复制
class NamedPipe
{
public:
    NamedPipe(const string &path) : _fifo_path(path)
    {
        int res = mkfifo(path.c_str(), 0666);
        if (res != 0)
        {
            perror("mkfifo");
        }
    }
    ~NamedPipe()
    {
        int res = unlink(_fifo_path.c_str());
        if (res != 0)
        {
            perror("unlink");
        }
    }
private:
    const string _fifo_path;
};

我的主函数直接构建对象即可:

代码语言:javascript
代码运行次数:0
运行
复制
NamedPipe myfifo(comm_path);

管道会自动释放

这里继续在类里面添加其他功能

代码语言:javascript
代码运行次数:0
运行
复制
const string comm_path = "./myfifo";

#define DefaultFd -1
#define Creater 1
#define User 2
#define Read O_RDONLY
#define Write O_WRONLY
class NamedPipe
{
private:
    bool OpenNamedPipe(int mode)
    {
        _fd = open(_fifo_path.c_str(), mode);
        if (_fd < 0)
            return false;
        return true;
    }

public:
    NamedPipe(const string &path, int who) : _fifo_path(path), _id(who), _fd(DefaultFd)
    {
        if (_id == Creater)
        {
            int res = mkfifo(path.c_str(), 0666);
            if (res != 0)
            {
                perror("mkfifo");
            }
        }
    }
    bool OpenForRead()
    {
        OpenNamedPipe(Read);
    }
    bool OpenForWrite()
    {
        OpenNamedPipe(Write);
    }
    ~NamedPipe()
    {
        if (_id == Creater)
        {
            int res = unlink(_fifo_path.c_str());
            if (res != 0)
            {
                perror("unlink");
            }
        }
        if(_fd!=DefaultFd) close(_fd);
    }

private:
    const string _fifo_path;
    int _id;
    int _fd;
};

隐藏打开文件的函数操作,定义宏常量Read和Write两种打开文件方式的mode,初始化传递执行者,是创建者还是使用者来决定构建的时候是否还要再进行管道文件创造

代码语言:javascript
代码运行次数:0
运行
复制
int ReadNamedPipe(string * out)//输出型参数,输入型const& 输入输出型&
    {
        char buffer[BaseSize];
        int n = read(_fd,buffer,sizeof(buffer));
        if(n>0)
        {
            buffer[n]=0;
            *out = buffer;
        }
        
        return n;
    }
    int WriteNamedPipe(const string & in)
    {
        write(_fd,in.c_str(),in.size());
    }
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2025-03-13,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 命名管道
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档