当前位置: 首页 > news >正文

Linux操作系统4-进程间通信5(共享内存实现两个进程通信)

上篇文章:Linux操作系统4-进程间通信4(共享内存原理,创建,查看,命令)-CSDN博客

本篇Gitee仓库:myLerningCode/l24 · 橘子真甜/Linux操作系统与网络编程学习 - 码云 - 开源中国 (gitee.com)

本篇重点:进程关联与去关联共享内存,实现两个进程通信

一. 进程关联共享内存

        两个进程想要通过共享内存进行通信,在创建共享内存之后还需要将这个共享内存与我们的进程相关联。

 进程关联共享内存的系统调用如下:

//头文件
#include <sys/types.h>
#include <sys/shm.h>

//函数原型
void* shmat(int shmid, const void* shmarr, int shmflg)

//参数
shimid 想要关联共享内存的id
shmarr 想要关联共享内存的地址空间,一般设置为NULL
shmflg 一般设置为0

//返回值
返回共享内存通过页表映射到进程地址空间的起始地址

二. 进程去关联共享内存 

        在我们使用完了这段共享内存后,如果想要让其他进程进行通信。可以让当前的进程与共享内存去关联

系统调用如下:

//头文件
#include <sys/types.h>
#include <sys/shm.h>

int shmdt(const void* shmaddr);

shmaddr 我们关联共享内存时候获取的地址

返回值:成功返回0,失败返回-1

三. 删除共享内存 

        在上篇文章中,我们使用命令 ipcs -m 查看共享内存。使用命令 ipcrm -m 删除共享内存

我们也可以通过系统调用来删除共享内存

系统调用如下:

#include <sys/ipc.h>
#include <sys/shm.h>

//用于对共享内存的控制(包含删除)
int shmctl(int shmid, int cmd, struct shmid_ds *buf)

cmd:传入的二进制标志位
IPC_RMID:立即移除对应的共享内存

buf:删除的话直接设置为NULL即可

返回值,失败返回-1,错误码被设置

四. 实现通信 

4.1 common.hpp

        这个头文件用于实现创建,关联,去关联,删除共享内存的代码。之后只需要让server.cpp和client.cpp去调用代码即可。

创建共享内存:对于client和server来说,需要由不同的操作,server需要创建共享内存,并且存在共享内存需要出错返回,而client只需要获取共享内存的id即可

代码如下:

#include <iostream>

#include <cassert>
#include <cstring>
#include <cstdlib>

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <error.h>
#include <unistd.h>

// 当前的绝对路径用于获取key
#define PATHNAME "."
#define PROJ_ID 2025221
#define MAXSIZE 5

// 获取key值
key_t getKey()
{
    // 获取一个key
    key_t key = ftok(PATHNAME, PROJ_ID);
    assert(key != -1);
}

// 传入key,创建共享内存,并返回其id
// flag是为了区分不同的进程,创建共享内存的进程需要出错返回,而另一个进程则不需要
int getShmHelp(const key_t key, int flag)
{
    int sid = shmget(key, MAXSIZE, flag);
    assert(sid >= 0);
}

// 共享内存存在,需要出错返回,并且提供权限
int creatShm(key_t key)
{
    return getShmHelp(key, IPC_CREAT | IPC_EXCL | 0666);
}

// 存在共享内存,不做任何事,只是返回id
int getShm(key_t key)
{
    return getShmHelp(key, IPC_CREAT);
}

关联共享内存:传入id,关联共享内存,返回地址空间的起始地址即可

// 关联共享内存与进程
void *attachShm(int shmid)
{
    void *mem = shmat(shmid, nullptr, 0);
    if ((long long)mem == -1l) // 指针占8字节
    {
        std::cerr << "attachShm error" << errno << ":" << strerror(errno) << std::endl;
        exit(1);
    }
    return mem;
}

去关联共享内存:传入id,去关联即可

// 去关联共享内存
void detachShm(void *start)
{
    int n = shmdt(start);
    if (n == -1)
    {
        std::cerr << "detachShm error" << errno << ":" << strerror(errno) << std::endl;
        exit(2);
    }
}

 删除共享内存:传入id,删除即可

// 删除共享内存
void deleteShm(int shmid)
{
    int n = shmctl(shmid, IPC_RMID, nullptr);
    if (n == -1)
    {
        std::cerr << "deleteShm error" << errno << ":" << strerror(errno) << std::endl;
        exit(3);
    }
}

4.2 server.cpp

        共享内存通信让进程地址空间与共享内存通过页表进行了映射,所以我们只需要直接向字符串一样输出信息即可。

server端接收来自client端发送的消息

代码如下:

#include "common.hpp"
#include <iostream>
#include <string>

int main()
{
    // 1.创建共享内存
    int shmid = creatShm(getKey());
    std::cout << "server 创建共享内存成功" << std::endl;

    // 2.关联共享内存和进程地址空间的映射
    char *start = (char *)attachShm(shmid);
    std::cout << "server 关联共享内存成功" << std::endl;
    sleep(5);

    // 开始通信
    while (true)
    {
        // 服务端接收信息,由于有地址空间的映射。
        // 我们只需要和字符串一样接收即可
        std::cout << start << std::endl;
        sleep(1);
    }

    // 3.去关联共享内存与进程地址空间的映射
    detachShm(start);
    std::cout << "server 去关联共享内存成功" << std::endl;

    // 4.强制删除共享内存
    deleteShm(shmid);
    std::cout << "server 删除共享内存成功" << std::endl;

    return 0;
}

4.3 client.cpp 

client端无需删除共享内存

#include "common.hpp"
#include <iostream>

int main()
{
    // 获取共享内存id
    int shmid = getShm(getKey());
    std::cout << "client 获取共享内存id成功" << " sid为" << shmid << std::endl;

    // 关联共享内存
    char *start = (char *)attachShm(shmid);
    std::cout << "client 关联共享内存成功" << std::endl;

    // 开始通信
    const char *message = "你好server,我是client ...";
    int cnt = 0;
    while (true)
    {
        snprintf(start, MAXSZIE, "%s[client pid:%d][消息编号:%d]", start, getpid(), cnt++);
        sleep(1);
    }

    // 去关联共享内存
    detachShm(start);
    std::cout << "client 去关联共享内存成功" << std::endl;

    // client无需删除共享内存,server删除
    return 0;
}

五. 分析测试结果

        运行测试结果如下:

先运行server,在运行ckient。可以看到server端成功接收来自client的消息

 我们让client关闭

 关闭client端之后,server端仍然打印最后一条65编号的信息。

这说明共享内存没有同步和互斥功能,如果想要实现这个功能需要我们加锁

六. 共享内存的特定与优缺点

        6.1 优点

共享内存的优点就是

是所有进程间通信中最快的,只要我们一方写入数据,另外一方可以直接看到,大大减少数据的拷贝次数

 因为共享内存不需要缓冲区,所以速度快。这里我们对比共享内存与管道通信

管道通信:

共享内存通信

 

        在不考虑键盘输入,显示器输出的情况下。

管道文件至少需要拷贝数据4次,而共享内存只需要拷贝数据两次。

很明显。共享内存大大减少的数据的拷贝次数 

6.2 缺点 

        在上面的测试中我们知道,共享内存的缺点是:不支持同步和互斥功能,如果读取速度较快,会读取到已经读取了的数据

相关文章:

  • 【多模态处理篇二】【深度揭秘:DeepSeek视频理解之时空注意力机制解析】
  • 2025年华为手机解锁BL的方法
  • 函数指针和函数名在内存中是如何表示的
  • 计算机专业知识【揭开汇编的神秘面纱:从基础概念到实际应用】
  • VMware虚拟机手动安装VMware Tools
  • 合并区间(56)
  • [创业之路-321]:创新开拓思维和经营管理思维的比较
  • rkipc main.c 中 rk_param_init函数分析
  • vue3项目开发总结
  • Java 12~14 新特性
  • 力扣LeetCode: 2506 统计相似字符串对的数目
  • 围棋打谱应用软件设计制作
  • C++ Primer 容器库概述
  • RocketMq\Kafka如何保障消息不丢失?
  • 【微服务优化】ELK日志聚合与查询性能提升实战指南
  • 【多线程】线程安全
  • [LeetCode]day27 28. 找出字符串中第一个匹配项的下标
  • 音视频入门基础:RTP专题(10)——FFmpeg源码中,解析RTP header的实现
  • Docker仿真宇树狗GO1
  • Spring Security+JWT+Redis实现项目级前后端分离认证授权
  • 轿车追尾半挂车致3死1伤,事故调查报告:司机过分依赖巡航系统
  • 全国铁路五一假期累计发送1.51亿人次,多项运输指标创历史新高
  • 体坛联播|国米淘汰巴萨晋级欧冠决赛,申花击败梅州避免连败
  • 退休11年后,71岁四川厅官杨家卷被查
  • 新加坡总理黄循财领导人民行动党胜选,外交部回应
  • 沙发上躺赢又如何?告别冠军绝缘体的凯恩,要开始收割荣誉了