【Linux】进程间通信_3

简介: 【Linux】进程间通信_3

七、进程间通信

1. 进程间通信分类

命名管道

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

我们只需要使用 mkfifo 命令就可以在当前路径下创建命名管道。

Makefile

.PHONY:all
all:pipe_server pipe_client
pipe_server:PipeServer.cc
  g++ -o $@ $^ -std=c++11
pipe_client:PipeClient.cc
  g++ -o $@ $^ -std=c++11
.PHONY:clean
clean:
  rm -f pipe_server pipe_client

Comm.hpp

#ifndef __COMM_HPP__
#define __COMM_HPP__
#include <iostream>
#include <string>
#include <sys/types.h>
#include <sys/stat.h>
#include <cerrno>
#include <cstring>
#include <unistd.h>
#include <fcntl.h>
using namespace std;
#define Mode 0666
#define Path "fifo"
// 命名管道类
class Fifo
{
public:
    Fifo(const string& path)
    : _path(path)
    {
        umask(0);
        // 创建命名管道
        int n = mkfifo(_path.c_str(), Mode);
        if (n == 0)
        {
            cout << "mkfifo success" << endl;
        }
        else
        {
            cout << "mkfifo failed, error : " << errno << "errstring : " << strerror(errno) << endl;
        }
    }
    ~Fifo()
    {
        // 删除命名管道
        int n = unlink(_path.c_str());
        if (n == 0)
        {
            cout << "remove " << _path << " success" << endl;
        }
        else
        {
            cout << "remove failed, error : " << errno << "errstring : " << strerror(errno) << endl;
        }
    }
private:
    // 文件路径 + 文件名
    string _path;
};
#endif

pipe_server(服务端接收消息):

#include "Comm.hpp"
#include <unistd.h>
int main()
{
    // 创建命名管道
    Fifo fifo(Path);
    // 打开命名管道文件
    int rfd = open(Path, O_RDONLY);
    if (rfd < 0)
    {
        cerr << "open failed, errno: " << errno << " , errnostring: " << strerror(errno) << endl;
        return 1;
    }
    char buffer[1024];
    // 循环读取管道数据
    while (true)
    {
        ssize_t n = read(rfd, buffer, sizeof(buffer) - 1);
        if (n > 0)
        {
            buffer[n] = 0;
            cout << "client says: " << buffer << endl;
        }
        else if (n == 0)
        {
            cout << "client quie, me too" << endl;
            break;
        }
        else
        {
            cerr << "read failed, errno: " << errno << " , errnostring: " << strerror(errno) << endl;
            break;
        }
    }
    // 关闭命名管道文件
    close(rfd);
    return 0;
}

pipe_client(客户端输入消息):

#include "Comm.hpp"
int main()
{
    // 以写的方式打开命名管道文件
    int rfd = open(Path, O_WRONLY);
    if (rfd < 0)
    {
        cerr << "open failed, errno: " << errno << " , errnostring: " << strerror(errno) << endl;
        return 1;
    }
    string inbuffer;
    // 循环读取用户输入并写入管道
    while (true)
    {
        cout << "Please Enter Your Message# ";
        getline(cin, inbuffer);
        if (inbuffer == "quit") break;
        ssize_t n = write(rfd, inbuffer.c_str(), inbuffer.size());
        if (n < 0)
        {
            cerr << "write failed, errno: " << errno << " , errnostring: " << strerror(errno) << endl;
        }
    }
    // 关闭管道文件
    close(rfd);
    return 0;
}

结果:


未完待续

目录
相关文章
|
22小时前
|
监控 Linux 应用服务中间件
探索Linux中的`ps`命令:进程监控与分析的利器
探索Linux中的`ps`命令:进程监控与分析的利器
|
2天前
|
存储 Shell Linux
Linux进程概念(下)
本文详细的介绍了环境变量和进程空间的概念及其相关的知识。
11 0
Linux进程概念(下)
|
1天前
|
监控 Linux
深入了解Linux的pmap命令:进程内存映射的利器
`pmap`是Linux下分析进程内存映射的工具,显示内存区域、权限、大小等信息。通过`/proc/[pid]/maps`获取数据,特点包括详细、实时和灵活。参数如`-x`显示扩展信息,`-d`显示设备。示例:`pmap -x 1234`查看进程1234的映射。注意权限、实时性和准确性。结合其他工具定期监控,排查内存问题。
|
1天前
|
网络协议 Linux Shell
技术笔记:Linux中的两种守护进程standalone和xinetd
技术笔记:Linux中的两种守护进程standalone和xinetd
|
1天前
|
算法 Linux 编译器
技术笔记:LINUX2.6.32下的进程分析
技术笔记:LINUX2.6.32下的进程分析
|
1天前
|
监控 网络协议 关系型数据库
如何在Linux中查看正在运行的进程以及过滤特定端口和进程名称
如何在Linux中查看正在运行的进程以及过滤特定端口和进程名称
4 0
|
2天前
|
Linux
linux指令按端口查找和杀死进程
linux指令按端口查找和杀死进程
10 0
|
2天前
|
Unix Linux 调度
一篇文章讲明白linux僵死进程
一篇文章讲明白linux僵死进程
|
2天前
|
消息中间件 负载均衡 Linux
【linux】匿名管道|进程池
【linux】匿名管道|进程池
4 0
|
2天前
|
Shell Linux
【linux】进程替换的应用|shell解释器的实现
【linux】进程替换的应用|shell解释器的实现
7 0