Linux——进程间通信(共享内存)

目录

system V共享内存

​编辑

共享内存函数

共享内存的建立过程

shmget函数

shmctl函数

shmat函数

shmdt函数

实例代码

共享内存的特点


system V共享内存

共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间(即内存通过某种映射关系连接到了进程使得进程能够访问这段内存),这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据。

共享内存函数

我们之前的文章介绍的管道通信本质是基于文件的,OS没有做过多的设计工作。

而system V进程间通信是OS特地设计的通信方式,是为了想尽一切办法让不同的进程看到同一份资源。(具体的方法就是将物理内存映射到进程的地址空间中,本质就是修改页表,在虚拟地址空间中开辟空间,用的是系统调用接口)

共享内存的建立过程

1、申请共享内存

2、共享内存挂接到地址空间

3、去关联共享内存

4、释放共享内存

前两点式通信过程,后两点是释放资源的过程

shmget函数

功能:用来创建共享内存

原型

        int shmget(key_t key, size_t size, int shmflg);

参数

        key:这个共享内存段名字

        size:共享内存大小

        shmflg:由九个权限标志构成,它们的用法和创建文件时使用的mode模式标志是一样                          的

返回值:成功返回一个非负整数,即该共享内存段的标识码;失败返回-1

 //comm.h                                                                                                                                                                                     1 #ifndef _COMM_H_ 2 #define _ COMM_H_3 #include <stdio.h>4 #include <sys/types.h>5 #include <sys/ipc.h>6 #include <sys/shm.h>7 8 #define PATHNAME "."9 #define PROJ_ID 0x666610 #define SIZE 409611 12 #endif                                                                                                                                                                                                       
    //sever.c1 #include "comm.h"2 #include <unistd.h>3 int main()4 {5   key_t k = ftok(PATHNAME, PROJ_ID);6   if(k < 0)7   {8     perror("ftok");9     return 1;10   }11   printf("%x\n", k);12   int shmid = shmget(k, SIZE, IPC_CREAT|IPC_EXCL|0644);13   if(shmid < 0)14   {15     perror("shmget");16     return 2;17   }18  19   return 0;20 }

运行这一段代码,便能成功创建一块共享内存,可以使用ipcs -m命令查看

shm的生命周期是随内核的,进程不主动删除或者用命令删除,共享内存就一直存在,直到关机重启。

释放这一块空间则是用命名ipcrm -m [shmid]

当然这块内存的释放也有函数可以使用,没必要在命令行进行这么麻烦的操作

shmctl函数

功能:用于控制共享内存

原型

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

参数

        shmid:由shmget返回的共享内存标识码

        cmd:将要采取的动作(有三个可取值)

        buf:指向一个保存着共享内存的模式状态和访问权限的数据结构

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

在上面创建的server.c中结尾加上这一段

sleep(10);shmctl(shm, IPC_RMID, NULL);sleep(10);    

运行这一段代码,可以看到共享内存一开始被创建,然后过了一会就被删除了

至此我们就完成了共享内存的创建与释放,那么接下来要解决的问题就是如何进行映射了

shmat函数

功能:将共享内存段连接到进程地址空间

原型

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

参数

        shmid:共享内存标识

        shmaddr:指定连接的地址

        shmflg:它的两个可能取值是SHM_RND和SHM_RDONLY

返回值:成功返回一个指针,指向共享内存的第一个节;失败返回-1

    1 #include "comm.h"2 #include <unistd.h>3 #include <string.h>4 #include <sys/types.h>5 #include <sys/ipc.h>6 #include <sys/shm.h>7 int main()8 {9   key_t _key = ftok(PATHNAME, PROJ_ID);10   if(_key < 0)11   {12     perror("ftok");13     return -1;14   }15   printf("%x\n",_key);16   int shmid = 0;17   if((shmid = shmget(_key, SIZE, IPC_CREAT|IPC_EXCL|0644)) < 0)18   {19     perror("shmget");20     return -2;21   }22 23   printf("shmid: %d\n", shmid);24 25   printf("attach begin!\n");26   sleep(3);27   char* mem = shmat(shmid, NULL, 0);28   if(mem == (void*)-1)29   {30     perror("shmat");31     return 3;32   }                                                                                             33   printf("attach end!\n");34   sleep(3);35 36   shmctl(shmid, IPC_RMID, NULL);37 38   return 0;39 }

说明:

shmaddr为NULL,核心自动选择一个地址

shmaddr不为NULL且shmflg无SHM_RND标识,则以shmaddr为连接地址

shmaddr不为NULL且shmflg设置了SHM_RND标记,则连接的地址会自动向下调整为SHML BA的整数倍。公式:shmaddr - (shmaddr % SHML BA)

shmflg=SHM_RDONLY,表示连接操作用来只读共享内存

shmdt函数

功能:将共享内存段与当前进程脱离

原型

        int shmdt(const void *shmaddr);

参数

        shmaddr:由shmat所返回的指针

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

注意:将共享内存段与当前进程脱离不等于删除共享内存段

取消关联即在上面那段代码中的释放共享内存前加入

shmdt(mem);

即可取消进程与共享内存间的关联

实例代码
//comm.h1 #ifndef _COMM_H_ 2 #define _ COMM_H_3 #include <stdio.h>4 #include <sys/types.h>5 #include <sys/ipc.h>6 #include <sys/shm.h>7 8 #define PATHNAME "."9 #define PROJ_ID 0x666610 #define SIZE 409611 12 #endif        
//server.c1 #include "comm.h"2 #include <unistd.h>3 int main()4 {5   key_t k = ftok(PATHNAME, PROJ_ID);6   if(k < 0)7   {8     perror("ftok");9     return 1;10   }11   printf("%x\n", k);12   int shmid = shmget(k, SIZE, IPC_CREAT|IPC_EXCL|0644);13   if(shmid < 0)14   {15     perror("shmget");16     return 2;17   }18 19   printf("shmid: %d\n", shmid);20 21   sleep(5);22   char* mem = shmat(shmid, NULL, 0);23   while(1)24   {25   printf("client msg# %s\n", mem);26   sleep(1);                                                                                     27   }28   shmdt(mem);29 30   shmctl(shmid, IPC_RMID, NULL);31 32   return 0;33 }34 
//client.c1 #include "comm.h"2 #include <unistd.h>3 #include <string.h>4 int main()5 {6   key_t k = ftok(PATHNAME, PROJ_ID);7   if(k < 0)8   {9     perror("ftor");10     return 1;11   }12   int shmid = shmget(k, SIZE,0644);13   if(shmid < 0)14   {15     perror("shmget");16     return 2;17   }18 19   char* mem = shmat(shmid, NULL, 0);20 21   int i = 0;                                                                                    22   while(1)23   {24     mem[i] = 'A' + i;25     sleep(1);26     i++;27     mem[i] = 0;28   }29 30   shmdt(mem);31 32   return 0;33 }

可以看到我们实现了client端向server端通过共享内存进行信息的传输

共享内存的特点

1、共享内存是所有进程通信中速度最快的,因为它没有使用OS接口,拷贝次数减少

2、不提供任何保护机制(同步与互斥)

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

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

相关文章

OpenCV-28 全局二值化

一、形态学概念 什么是形态学&#xff1f; 1&#xff09;指一系列处理图像型状特征的图像处理技术 2&#xff09;形态学的基本思想是利用一直特殊的结构元&#xff08;本质上是卷积核&#xff0c;且这个卷积核的值只有1和0&#xff09;来测量或提取输入图像中相应的型状或特…

深入了解Go语言中的HTTP代理处理机制

Go语言&#xff0c;也被称为Golang&#xff0c;自2009年诞生以来&#xff0c;凭借其简洁、高效的特点&#xff0c;逐渐在系统编程、Web开发等多个领域崭露头角。在Web开发中&#xff0c;HTTP代理处理是常见的需求之一。本文将深入探讨Go语言中的HTTP代理处理机制。 HTTP代理是…

【Java 数据结构】LinkedList与链表

LinkedList与链表 1. ArrayList的缺陷2. 链表2.1 链表的概念及结构2.2 链表的实现 3. LinkedList的模拟实现4.LinkedList的使用4.1 什么是LinkedList4.2LinkedList的使用 5. ArrayList和LinkedList的区别 1. ArrayList的缺陷 上节课已经熟悉了ArrayList的使用&#xff0c;并且…

【linux】Linux项目自动化构建工具-make/Makefile

make/makefile 背景 会不会写makefile&#xff0c;从一个侧面说明了一个人是否具备完成大型工程的能力 一个工程中的源文件不计数&#xff0c;其按类型、功能、模块分别放在若干个目录中&#xff0c;makefile定义了一系列的规则来指定&#xff0c;哪些文件需要先编译&#xff0…

Java笔记 --- 二、Stream流

二、Stream流 结合Lambda表达式&#xff0c;简化集合、数组的操作 获取Stream流对象 单列集合获取Stream流 双列集合获取Stream流 数组获取Stream流 一堆零散的数据获取Stream流 Stream流的静态方法of的形参是一个可变参数&#xff0c;可以传递零散数据&#xff0c;也可以传递…

【计算机网络】IP协议及动态路由算法

对应代码包传送门 IP协议及动态路由算法代码包及思科模拟器资料说明 相关文章 【计算机网络】中小型校园网构建与配置 【计算机网络】Socket通信编程与传输协议分析 【计算机网络】网络应用通信基本原理 目的&#xff1a; 1、掌握IP协议&#xff0c;IP分片&#xff0c;DH…

Ubuntu上 RISC-V64 Jemalloc 编译补丁(修复无法链接问题)

它的问题跟这个是相同的&#xff1b; RISC-V平台 std::atomic&#xff1c;T&#xff1e; 编译失败问题解决-CSDN博客 区别是自己写的代码&#xff0c;能改掉&#xff0c;但是 Jemalloc 编译好的静态库。 比如&#xff1a;我们是在其它平台上面交叉编译的 RISC-V程序&#xff0…

获取el-select的label

获取el-select的label 使用click.navite 在 el-option 中添加 clik.native 在方法的传参中传入自己需要的内容 这个方法同样适用于 element-plus <template><el-select v-model"selectValue"><el-option v-for"item in options" :key&q…

04. OSPF

文章目录 一. 初识OSPF1.1. OSPF概述1.2. OSPF特性1.3. OSPF的专业术语1.4. OSPF维护的3张表1.5. OSPF报文类型1.6. OSPF的邻居状态 二. 实验题2.1. 实验1&#xff1a;点到点链路上的OSPF2.1.1. 实验目的2.1.2. 实验拓扑图2.1.3. 实验步骤&#xff08;1&#xff09;IP地址配置&…

机器学习面试题总结

1.请简要介绍下SVM。 SVM,全称是support vector machine,中文名叫支持向量机。SVM是一个面向数据的分类算法,它的目标是为确定一个分类超平面,从而将不同的数据分隔开。 扩展: 支持向量机学习方法包括构建由简至繁的模型:线性可分支持向量机、线性支持向量机及非线性支…

【无标题】Gateway API 实践之(五)FSM Gateway 的会话保持功能

网关的会话保持功能是一种网络技术&#xff0c;旨在确保用户的连续请求在一段时间内被定向到同一台后端服务器。这种功能在需要保持用户状态或进行连续交互的场景中特别重要&#xff0c;例如在维护在线购物车、保持用户登录状态或处理多步骤事务时。 会话保持通过提供一致的用…

判断素数的程序

判断一个数是否为素数的常见算法是试除法。素数&#xff08;质数&#xff09;是指只能被1和自身整除的正整数&#xff0c;因此可以通过逐个尝试除以小于该数的所有正整数来判断一个数是否为素数。 以下是用于判断素数的简单算法的示例代码&#xff1a; #include <stdio.h&…

中科大--高级数据库期末试卷

2023 秋一、单项选择题(10 分)二、判断题。正确的打√,错误的打(20 分)三、(10 分)四、(20 分)五、(20 分)六、(20 分)2021 秋2023 秋 中 国 科 学 技 术 大 学 2023—2024 学年第一学期考试试卷 考议形式:闭卷考试,允许使用计算器 考试科目:高级数据库系统…

【计算机网络】-换种方式思考万维网的发展

万维网WWW&#xff08;world wild web&#xff09;并非是某种特殊的计算机网络。万维网是一个大规模的、联机式的信息储藏所。 从应用的反方向去思考WEB在发展中遇到的问题 &#xff08;1&#xff09;如何标志发布在整个因特网上的文档&#xff1f; &#xff08;2&#xff0…

java判空工具类 java判断对象属性是否全部为空

java判空工具类 java判断参数是否为空工具类 java判断对象属性是否全部为空 。 可以判断字符串数组&#xff0c;字符串类型&#xff0c;List对象集合&#xff0c;Map集合map<String,object>&#xff0c;map和多个String参数&#xff0c;对象类型&#xff0c;是否为空。…

QT案例 Qt调用WIMGAPI库,读取/修改/装载WIM、ESD格式的镜像文件

WIMGAPI库是 Windows 操作系统中的一个应用程序编程接口&#xff08;API&#xff09;可用于创建&#xff0c;编辑&#xff0c;提取浏览和部署 WIM、ESD格式的镜像文件。 这里只简单演示在Qt Creator&#xff0c; Mscv2017 Debug&#xff0c;WINDOWS 开发环境下的调用完整实例。…

数据结构C++队列(数组模拟)

队列也是比较简单的数据结构了&#xff0c;队列的特点是先进先出 下面代码中hh是队头&#xff0c;tt是队尾。 默认是从队尾插入数据&#xff0c;队头弹出数据。 代码中的数据结构可以使用这图片来解释&#xff0c;整个区间是数组q。hh和tt分别控制队头和队尾。 例题&#x…

前端-打卡每日面试题-数据类型(2024.1.26)

一、分类 在JavaScript中&#xff0c;数据类型可以分为两大类&#xff1a;原始数据类型&#xff08;Primitive Types&#xff09;和引用数据类型&#xff08;Reference Types&#xff09;。 1. 原始数据类型&#xff08;Primitive Types&#xff09;: Number&#xff08;数字…

AI作画工具 stable-diffusion-webui 一键安装工具(A1111-Web-UI-Installer)

安装 下载最新版本确保你的 NVIDIA 显卡驱动程序是最新的&#xff08;起码不能太老&#xff09;启动安装程序在欢迎屏幕上单击下一步在屏幕上&#xff0c;选择要安装的内容如果你已经安装了 Python 3.10 和 Git&#xff0c;那么可以取消选中如果你不知道这些是什么&#xff0c…

C++(5) 构造函数、析构函数和封装案例

文章目录 构造函数、析构函数和封装案例1. 构造函数和析构函数1.1 概述1.2 案例1.3 编译器自行提供构造函数1.4 拷贝构造函数和特殊用法1.5 构造函数总结和要求 2. 封装案例2.1 封装的终极奥义2.2 封装一个 Cube 立方体 构造函数、析构函数和封装案例 1. 构造函数和析构函数 …