linux篇【9】:进程间通信(共享内存)——<后序>

news2024/11/25 18:50:52

目录

一.system V共享内存——先让不同的进程看到同一份资源

1.共享内存原理

监控共享内存脚本

 2.创建/获取 共享内存接口—shmget函数(shared memory get)

3.参数key解释

(1)共享内存存在哪里?

(2)第一个参数key的解释

(3)key一般由用户提供

(4)函数ftok获取key

(5)查看共享内存的命令 ipcs -m,删除共享内存 ipcrm -m (要删的共享内存的shmid)

3.删除共享内存的系统接口 shmctl

4.关联函数shmat、去关联函数shmat

5.我怎么知道,这个共享内存属于存在还是不存在?


一.system V共享内存——先让不同的进程看到同一份资源

进程问通信的前提是:先让不同的进程,看到同一份资源!

1.共享内存原理

原理:创建共享内存和分别把共享内存挂接到各自的进程上下文中:通过一些系统接口在物理内存中创建一块空间,通过进程对应的页表映射到进程1和进程2各自的地址空间中,然后把这个两个进程的进程地址空间的地址都返回给用户

监控共享内存脚本

while :; do ipcs -m; sleep 1; echo "############################";done

 2.创建/获取 共享内存接口—shmget函数(shared memory get)

①创建共享内存——②删除共享内存
③关联共享内存——④去关联共享内存
这些操作都是OS内部帮我们做,跟进程关联

shmget函数:

创建一个System V 级别的共享内存段
int shmget(key_t key, size_t size, int shmflg);        

返回值:成功返回共享内存标识符shmid;错误返回-1

key:共享内存内核的唯一值(用函数ftok获取key)详解看 3.—>(2)(3)

size:创建的共享内存大小。(建议设置成为页(4KB)的整数倍,因为操作系统和磁盘IO时基本单位是4KB)

shmflg的选项:

IPC_CREAT:创建共享内存,如果已经存在就获取之,不存在就创建之
IPC_ EXCL:不单独使用,必须和IPC_CREAT配合。如果不存在指定的共享内存,创建之;如果存在了,出错返回
IPC_CREAT和IPC_CREAT配合:可以保证,如果shmget函数调用成功—— 一定是一个全新的share memory!(即:如果只是使用IPC_CREAT返回后无法区分是新创建的还是原来就存在的,如果配合使用返回成功,就是新的共享内存;不成功就是原来已经存在的

shmflg | 0666 :支持异或上权限

3.参数key解释

(1)共享内存存在哪里?

——内核中,内核会给我们维护共享内存的结构!共享内存也要被管理起来! !先描述,再组织!

(2)第一个参数key的解释

想表示存在不存在,就要先有方法标识共享内存的唯一性! !——用key标识
共享内存要被管理
struct shmid_ ds{} 中包含一个结构体 struct ipc_ perm ,这个结构体 struct ipc_ perm 中包含一个成员值叫 key (shmget) (key的作用:共享内存的唯一值! !这个key一般由用户提供)

(3)key一般由用户提供 解释

进程问通信的前提是:先让不同的进程看到同一份资源。如何保证让不同的进程看到同一份共享内存?——做法是:让他们拥有同一个key即可!(用函数ftok获取key)

命名管道——>约定使用同一个文件
共享内存——>约定好使用同一个唯一key,来进行通信的! !
当我们运行完毕创建全新的共享内存的代码后(进程退出),但是第二(n)次的时候,该代码无法运行,告诉我们file存在,即:共享内存是存在的!
systen-V下的共享内存,生命周期是随内核的! !
如果不显示的删除,只能通过kernel (os) 重启来解决!

(4)函数ftok获取key

作用:把一个文件路径和项目标识符转化为一个具有唯一性的数字。

 key_t ftok(const char *pathname, int proj_id);

返回值:成功返回生成的key;失败返回-1

例如:

#define PATH_NAME "/home/whb/104"

#define PROJ_ID 0x14 

key_t key = ftok(PATH_NAME, PROJ_ID)        PROJ_ID随便设一个

(5)查看共享内存的命令 ipcs -m,删除共享内存 ipcrm -m (要删的共享内存的shmid)

perm:权限; nattch:挂接进程数

3.删除共享内存的系统接口 shmctl

int shmctl(int shmid, int cmd, struct shmid_ds *buf);

shmid:共享内存标识符,表示要删哪一个共享内存。

cmd:选项有常用的IPC_RMID(立即删除,此时buf传nullptr即可删除)(还有IPC_SET设置,IPC_STAT拷贝选项)

例如 shmctl(shmid, IPC_RMID, nullptr);

4.关联函数shmat

shmat (attach-附上),shmdt(detach-拆卸)

void *shmat(int shmid, const void *shmaddr, int shmflg);

shmid:共享内存标识符。shmaddr:设置在地址空间哪个位置,现在设置为nullptr。shmflg:读写,现在设置为0

返回值:成功返回挂接的共享内存地址,出错返回(void*)-1。

返回值和malloc一个意思,你怎么用malloc的空间,你就怎么用共享内存的空间! ( 肯定不能free)

例如:char *str = (char *)shmat(shmid, nullptr, 0);

————————————————————————

shmdt(detach-拆卸)

int shmdt(const void *shmaddr);

shmaddr:就传shmat的返回值。就能去关联

5.我怎么知道,这个共享内存属于存在还是不存在?

只需拿曾经设置过的key和你要设置的key作对比,来确认共享内存是否存在。

6.让进程看到同一份资源的操作

Log.hpp

#pragma once

#include <iostream>
#include <ctime>

std::ostream &Log()
{
    std::cout << "Fot Debug |" << " timestamp: " << (uint64_t)time(nullptr) << " | ";
    return std::cout;
}

Comm.hpp

(.hpp是声明和定义写在一起的文件)

#pragma once

#include <iostream>
#include <cstring>
#include <cstdlib>
#include <cerrno>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>

#define PATH_NAME "/home/whb/104"
#define PROJ_ID 0x14
#define MEM_SIZE 4096

key_t CreateKey()
{
    key_t key = ftok(PATH_NAME, PROJ_ID);
    if(key < 0)
    {
        std::cerr <<"ftok: "<< strerror(errno) << std::endl;
        exit(1);
    }
    return key;
}

 Makefile

​
.PHONY:all
all: IpcShmCli IpcShmSer

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

.PHONY:clean
clean:
	rm -f IpcShmCli IpcShmSer

​

lpcShmCli.cc

#include "Comm.hpp"
#include "Log.hpp"

#include <unistd.h>

using namespace std;
// 充当使用共享内存的角色
int main()
{
    // 创建相同的key值
    key_t key = CreateKey();
    Log() << "key: " << key << "\n";

    // 获取共享内存
    int shmid = shmget(key, MEM_SIZE, IPC_CREAT);
    if (shmid < 0)
    {
        Log() << "shmget: " << strerror(errno) << "\n";
        return 2;
    }

    // 挂接
    char *str = (char*)shmat(shmid, nullptr, 0);

    // 用它
    sleep(5);

    // 去关联
    shmdt(str);

    return 0;
}

IpcShmSer.cc

#include "Comm.hpp"
#include "Log.hpp"

#include <unistd.h>

using namespace std;

// 我想创建全新的共享内存
const int flags = IPC_CREAT | IPC_EXCL;

// 充当使用共享内存的角色
int main()
{
    key_t key = CreateKey();
    Log() << "key: " << key << "\n";

    Log() << "create share memory begin\n";
    int shmid = shmget(key, MEM_SIZE, flags | 0666);
    if (shmid < 0)
    {
        Log() << "shmget: " << strerror(errno) << "\n";
        return 2;
    }
    Log() << "create shm success, shmid: " << shmid << "\n";
    sleep(5);

    // 1. 将共享内存和自己的进程产生关联attach
    char *str = (char *)shmat(shmid, nullptr, 0);
    Log() << "attach shm : " << shmid << " success\n";

    sleep(5);
    // 用它

    // 2. 去关联
    shmdt(str);
    Log() << "detach shm : " << shmid << " success\n";
    sleep(5);

    // 删它
    shmctl(shmid, IPC_RMID, nullptr);

    Log() << "delete shm : " << shmid << " success\n";

    sleep(5);
    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/4502.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Spring Boot DTO 验证示例

在本教程中&#xff0c;我们将学习如何使用 Hibernate 验证器验证 Spring 启动 REST API DTO 请求。 在Java中&#xff0c;Java Bean Validation框架已经成为处理Java项目中验证的事实标准。 JSR 380 是用于 Bean 验证的 Java API 规范&#xff0c;它使用 NotNull、Min 和 Ma…

深入理解Java虚拟机

Java Virtual MachineJVM内存模型类加载器沙箱安全机制Native 和 方法区栈、队列、堆三种JVM垃圾回收一次完整的GCJVM内存模型 .class文件在进入类加载器后&#xff0c;进行加载-连接-初始化 类加载器 public class User {private String name;private Integer age;public st…

什么是浏览器的缓存机制

先来粗略的概念&#xff1a; 什么是浏览器的缓存机制 浏览器的缓存机制就是把一个请求过的web资源&#xff08;例如&#xff1a;html页面、图片、js、数据等&#xff09;拷贝一份副本储存在浏览器中&#xff1b;缓存会根据进来的请求保存输出内容的副本&#xff0c;当下一个请求…

frp内网穿透并实现开机自启动

frp配置内网穿透、ssh远程连接、systemctl自启动 1.服务器端 VPS 配置内网穿透 修改frps.ini文件&#xff1a; # frps.ini[common]bind_port 7000 启动frps&#xff1a; ./frps -c ./frps.ini 2.客户端配置 修改 frpc.ini 文件&#xff0c;假设 frps 所在服务器的公网 IP 为…

【C++笔试强训】第二十天

&#x1f387;C笔试强训 博客主页&#xff1a;一起去看日落吗分享博主的C刷题日常&#xff0c;大家一起学习博主的能力有限&#xff0c;出现错误希望大家不吝赐教分享给大家一句我很喜欢的话&#xff1a;夜色难免微凉&#xff0c;前方必有曙光 &#x1f31e;。 &#x1f4a6;&a…

初识C++(五)

简述&#xff1a;初识C章节最后一节啦 整体感觉就是C像是C的补充和升级 以一种更简单的方式奔向普罗大众 从而也能使更多人接受编程 当然不是讲C简单 就是C像是从机器时代进入了电气时代 以更简单的操作实现更高的效率&#xff0c;这是我在接触C一周时的整体印象。 目录 auto关…

学习python第6天

函数 函数的作用&#xff1a; 函数是组织好的,可以重复使用的、用来实现单一功能的代码 函数的组成   数学函数 y 6 * x 9&#xff0c;x 是自变量&#xff0c;6 * x 9 是执行过程&#xff0c;y 是因变量&#xff0c;自变量 x 决定了因变量 y 的值。 那么&#xff0c;你…

Python美化桌面—自制桌面宠物

前言 嗨嗨&#xff0c;最近就喜欢搞一些花里胡哨的东西 这不就开始折腾我的电脑了吗 浅浅搞个桌面小挂件&#xff08;桌面宠物&#xff09; 前期准备 开发工具 Python版本&#xff1a;3.6.4 相关模块&#xff1a; PyQt5模块&#xff1b; 以及一些Python自带的模块。 …

花了一天的时间,地板式扫盲了vue3中所有API盲点

&#x1f4cd;前言 最近在一次理解vue项目的代码时&#xff0c;发现周一对好多API都不太熟悉。这间接导致的问题是&#xff0c;代码理解速度要比平常要慢很多。于是乎&#xff0c;赶忙把vue API的学习提上了日程。 在下面的文章中&#xff0c;将地板式地扫盲vue3文档中API模块…

MALUNet:一种多关注,轻量级的皮肤病变分割UNet

Tittle&#xff1a;MALUNet: A Multi-Attention and Light-weight UNet for Skin Lesion Segmentation 摘要 本篇文章提出的目的是为了缓解计算资源的不足&#xff0c;轻量化模型提出的。简单来说本文提出了4个模块。 1&#xff09;DGA&#xff1a;由扩展卷积和门控注意机制…

【数据结构初阶】复杂链表复制+带头双向循环链表+缓存级知识

我父亲曾将我高举过头顶&#xff0c;所以到哪我都不会觉得自己低人一等。 加油&#xff01;少年&#xff01; 文章目录一、复杂链表复制二、带头双向循环链表的实现2.1 简单介绍一哈2.2 结构的定义链表初始化2.3 开辟结点空间头插尾插头删尾删链表打印2.4 结点查找pos之前插入…

【读论文】DDcGAN

DDcGAN: A Dual-Discriminator ConditionalGenerative Adversarial Network for Multi-Resolution Image Fusion简介贡献提出的方法损失函数网络结构生成器结构辨别器结构MRI和PET图像融合处理思路公式损失函数处理过程训练总结参考论文&#xff1a; https://ieeexplore.ieee.o…

双11,用Python爬取4000多条笔记本电脑的销售数据并分析

双11&#xff0c;用Python爬取4000多条笔记本电脑的销售数据并分析 2022年&#xff0c;大部分电子产品的销量都在下滑&#xff0c;比如手机&#xff0c;一方面&#xff0c;产品的创新和提升不足&#xff0c;另一方面&#xff0c;大部分人更愿意把钱存起来&#xff0c;降低生活…

PolarDB for PostgreSQL透明加密

PolarDB for PostgreSQL透明加密简介透明加密TDE(Transparent Data Encryption)通过在数据库层执行静止数据加密&#xff0c;阻止可能的攻击者绕过数据库直接从存储读取敏感信息。经过数据库身份认证的应用和用户可以继续透明地访问数据&#xff0c;而尝试读取数据库文件中敏感…

基于BP神经网络的非线性函数拟合(一维高斯函数)研究-含Matlab代码

目录一、引言二、BP神经网络的结构与原理2.1 信息前向传播2.2 误差的反向传播过程三、基于BP神经网络的非线性函数拟合3.1 数据生成3.2 神经网络拟合结果四、参考文献五、Matlab代码获取一、引言 以生物学基础的BP网络为全局网络&#xff0c;于多变量函数逼近理论不同&#xf…

C语言之文件的使用(上)

目录 目录 前言 预备知识 一、为什么要使用文件&#xff1f; 二、什么是文件&#xff1f;文件有哪些分类&#xff1f; 1.程序文件 2.数据文件 三、文件名 四、对文件的操作 1.基础知识&#xff08;含文件指针的介绍&#xff09; 2.文件打开与关闭的相关函数 1.fope…

【前沿技术RPA】 RPA开发人员的日常

&#x1f40b;作者简介&#xff1a;博主是一位.Net开发者&#xff0c;同时也是RPA和低代码平台的践行者。 &#x1f42c;个人主页&#xff1a;会敲键盘的肘子 &#x1f430;系列专栏&#xff1a;UiPath &#x1f980;专栏简介&#xff1a;UiPath在传统的RPA&#xff08;Robotic…

C语言经典题目之青蛙跳台阶问题

目录 一、问题描述 二、问题分析 1.当n1时 2.当n2时 3.当n3时 4.n4&#xff0c;n5........nn时 三、代码实现 总结 一、问题描述 一只青蛙一次可以跳上 1 级台阶&#xff0c;也可以跳上2 级。求该青蛙跳上一个n 级的台阶总共有多少种跳法。 二、问题分析 青蛙跳台阶&a…

python的opencv操作记录(八)——小波变换

文章目录什么是小波从一个例子入手把例子再深化一下各种个样的小波基哈尔小波其他小波小波分解图像(二维)小波变换什么是小波 上一篇里提到了stft&#xff0c;短时傅里叶变换&#xff0c;是针对不稳定信号进行加窗来做每一个小窗口的频谱分析。然后一个一个的时间窗就可以理解…

STM32F103 UART4串口使用DMA接收不定长数据和DMA中断发送

一、前言 使用DMA通信的好处是&#xff0c;不占用单片机资源&#xff08;不像普通串口中断&#xff0c;发送一个字节触发一次中断&#xff0c;发送100个字节触发100次中断&#xff1b;接收一个字节触发一次中断&#xff0c;接收200个字节触发200次中断&#xff09;&#xff0c;…