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

进程的共享主存通信实验

进程的共享主存通信

【预备知识】

共享存储区为进程提供了直接通过主存进行通信的有效手段,不像消息缓冲机制那样需要系统提供缓冲,也不像pipe机制那样需要事先建立一个特殊文件,而是由通信双方直接访问某些共享虚拟储存空间。在Linux中,系统管理一组共享主存段控制块。通信进程在使用共享主存段以前,首先提出申请,系统为之分配存储空间并返回共享主存段标识号。一个共享段建立后,进程把它被附加到自己虚拟空间中。一个进程可以附加多个共享主存段。一个共享主存段一旦被附加到进程的虚拟空间后,对它的访问与其他虚拟地址的访问完全相同。但为了保证共享主存段数据的完整性,通信的进程之间要互斥地进行访问。当通信进程不再需要该共享主存段时,可使用命令将其与进程分离,从而使其从进程的虚空间删除。

  1. 共享主存段使用的数据结构

(1)共享主存段控制块(或共享主存段头)

每个共享主存段都有一个控制块,用来描述共享主存段的一些属性,共享主存段控制块定义在sys/shm.h中,其结构如下:

struct shmid_ds

{

struct ipc_perm shmperm;         / * 共享主存段访问控制结构 * /

int shmsegsz;                    / * 共享段以字节为单位的长度 * /

struct ptentry * shmptbl;        / * 共享页表始址 * /

ushort shmlpid;                  / * 最近执行共享段操作的进程标识 * /

ushort shmcpid;                  / * 创建共享段的进程标识 * /

ushort shmnattch;                / * 当前附件段号 * /

ushort shmcnattch;               / * 主存中的附加段号 * /

time_t shmatime;                 / * 最近一次附件操作的时间 * /

time_t shmdtime;                 / * 最近一次与进程分离操作的时间 * /

time_t shmctime;                 / * 最近一次修改时间 * /

}

为了便于管理,系统将维持的共享主存段组成一个表,共有SHMMNI=100个元素,其结构如下:

struct shmid_ds shmen[SHMMNI];   / * 共享段表 * /

其访问控制结构定义如下:

struct ipc_perm

   key_t key;

   ushort uid;              /* owner euid and egid * /

   ushort gid;

   ushort cuid;

  ushort cgid;

  ushort mode; /*lower 9 bits of shmflg*/

  ushort seq   /*sequence number*/

};

(2)共享主存段的数据结构

每个共享主存段都对应一个页表和允许的存取权限,结构如下;

struct shmptds

{

int shmspte;  /*开始也表项*/

int shmsflg;  /*对共享段的读/写权限*/

}

每个进程最多允许6个共享主存段(SHMSEG=6).

  1. 申请一个共享主存段

参与通信的进程,通信前要先申请一个共享主存段,若是第一次申请,则要为其分配一个主存空间及页表,并对共享主存区控制块进行初始化,申请共享主存段调用语法如下:

# include<sys/ipc.h>

# include<sys/shm.h>

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

其中,key为共享主存段的关键字,size是共享主存段字节长度,shmflg为创建和访问标志。

返回值:成功时,为与key值相关的共享主存段的标识号,且大小是页对齐;失败时,为-1。

如果key的值为IPC_PRIVATE,或不为IPC_PRIVATE,创建的共享段与key无关,关键字由系统分配。

Shmflg由如下成分组成:

.IPC_CREAT,创建一个新段。如果该标志没有设置,将查找与key相关的段,且该段允许用户访问。

.IPC_EXCL,与IPC_CREAT一起使用,确保创建一个新的共享段。若该段已经存在,出错。

低9位为三类用户的访问方式的定义。

  1. 将共享段附加到申请通信的进程空间

对于已申请通信所需的共享段,进程需把它附加到自己的虚拟空间后才能对其进行读写,将共享段附加到申请通信的进程空间的函数调用语法:

#include < sys/sem.h >

void shmat ( int shmid,coid * shmadd,nt shmflg);

这里shmid是进程调用shmget后返回的共享段标识号;shmadd是给出的应附加到进程虚空间的地址;shmflg为允许对共享段的访问方式。

返回值:成功时为附加到进程地址空间的虚地址,失败时为-1。

连接到进程虚空间的地址规则为:

.shmadd为0,则将该共享段附加到系统选择的进程的第一个可用地址之后。

.shmadd为非0,如果shmflg指定了SHM_RND标志,则将该共享段附加shmadd取整后指定的地址上。

.shmadd为非0,如果shmflg未指定SHM_RND标志,则将该共享段附加shmadd指定的地址上。通常将shmadd指定为0,由系统安排连接地址。

4.将共享段与进程之间解除链接

当进程不再需要共享段时,将其从它的地址空间拆下,调用语法为:

#include < sys/sem.h >

int shmdt( void * shmaddr );

其中,shmaddr是共享段在进程地址空间的虚地址。返回值为0。

5.对共享段进行控制

#include < sys/sem.h >

int shmctl(int shmid,int cmd,struct_ds * buf);

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

系统调用shmctl根据控制命令cmd对共享段进行控制。cmd可能的取值有:

. IPC_STAT,将有关共享段的信息复制到具有shmid_ds结构的buf中;

. IPC_SET,用于用户和用户组,或对其存取权限进行修改,要求用户必须为超级用户或拥有者;

. IPC_RMID,用于标志删除共享段。在共享段与所有进行分离时,实际进行删除;

. IPC_LOCK,在Linux环境下,将共享段锁在主存中;

. IPC_UNLOCK,在Linux环境下,允许将共享段锁换出主存。

编程实现例<一>

【任务】

编程实现一个是进程向共享段写信息的例子。

【程序】

#include  <sys/types.h>#include  <sys/ipc.h>#include  <sys/shm.h>#define SHMKEY 75#define  K  1024int shmid;main(){int i,*pint;char *addr;extern char *shmat();shmid=shmget(SHMKEY,16*K,0777|IPC_CREAT);/*申请创建一个16K大小的共享存储段,其标识号为SHMKEY*/addr=shmat(shmid,0,0);   /*缺省地址将共享段附加到系统指定位置*/printf(“addr ox%x\n”,addr);/*打印附加到进程地址空间的地址*/pint=(int *) addr;for (i=0;i<256;i++)*pint++=i;           /*向共享主存段写入0~255个数*/pint=(int *)addr;*pint=256;          /*共享段第一个字中写入信息长度256,以便接收进程读*/Pause();           /*暂停等待接收进程读*/}

修正后的代码:

【运行结果】

【分析】

第一个例子是一个进程向共享段写信息的例子,它使用了共享内存来实现进程间的通信。程序首先申请创建一个16K大小的共享存储段,标识号为SHMKEY,然后将共享段附加到系统指定位置,并打印附加到进程地址空间的地址。接着向共享主存段写入0~255个数,并在共享段第一个字中写入信息长度256,以便接收进程读取。最后暂停等待接收进程读

编程实现例<二>

【任务】

编程实现从共享段读信息的例子

【程序】

#include  <sys/types.h>#include  <sys/ipc.h>#include  <sys/shm.h>#define SHMKEY 75#define  K  1024int shmid;main(){int i,*pint;char *addr;extern char *shmat();shmid=shmget(SHMKEY,8*K,0777);  /*由SHMKEY得到共享段标识号*/addr=shmat(shmid,0,0);   /*将共享段与本进程相连*/pint=(int *) addr;while (*pint==0);       /*若共享区无信息时,在此等待*/for (i=0;i<256;i++)printf(“%d\n”,*pint++);}
修正后的代码:

【运行结果】

【分析】

第二个例子是一个从共享段读信息的例子,同样使用了共享内存来实现进程间的通信。程序首先通过SHMKEY得到共享段标识号,然后将共享段与本进程相连。接着程序进入一个while循环,在共享区无信息时,在此等待。当共享区有信息时,将信息读取出来并打印出来。

相关文章:

进程的共享主存通信实验

进程的共享主存通信 【预备知识】 共享存储区为进程提供了直接通过主存进行通信的有效手段&#xff0c;不像消息缓冲机制那样需要系统提供缓冲&#xff0c;也不像pipe机制那样需要事先建立一个特殊文件&#xff0c;而是由通信双方直接访问某些共享虚拟储存空间。在Linux中&…...

深度缓冲技术在AI去衣中的神奇作用

引言&#xff1a; 随着人工智能技术的飞速发展&#xff0c;其在图形处理和视觉领域的应用日益增多。AI去衣技术便是其中一个颇具争议但又技术上引人入胜的话题。今天&#xff0c;我们将深入探讨一项关键技术——深度缓冲&#xff08;Depth Buffering&#xff09;&#xff0c;它…...

能效?性能?一个关于Windows下使用openssl speed进行速度测试的诡异问题

问题描述 最近的某个软件用到了openssl&#xff0c;所以就想着测试一下速度。我的电脑是惠普的&#xff0c;CPU是AMD Ryzen 7 PRO 6850HS&#xff0c;系统是Win11。我使用openssl自带的speed测试加密/解密的速度&#xff0c;命令大致如下&#xff1a; openssl speed -evp aes…...

block性能考虑和线程安全

性能考虑 频繁地创建和销毁大量的 block 可能会对性能造成影响&#xff0c;特别是当这些 block 被拷贝到堆上时。同时&#xff0c;block 捕获大量数据时也会增加内存使用。 在讨论性能考虑时&#xff0c;主要关注的是 block 的创建、拷贝到堆上以及捕获变量的成本。以下是针对…...

没有公网ip,如何实现外网访问内网?

目前拨号上网是最广泛的上网方式&#xff0c;这种方式优点是价格便宜&#xff0c;缺点是没有固定公网ip&#xff0c;每次重新您拨号ip地址都会变。如果有一台服务器&#xff0c;需要实现外网访问&#xff0c;在没有固定公网ip的环境下&#xff0c;该如何实现呢&#xff1f;使用…...

Python中如何将小数转化为百分数进行输出

小数转化为百分数 Python中如何将小数转化为百分数进行输出基本概念使用字符串格式化1. 使用字符串格式化操作符 %2. 使用str.format()方法3. 使用f-string&#xff08;格式化字符串字面量&#xff09; **重点内容**&#xff1a;**无论是通过使用%格式化操作符、str.format()方…...

加入全球少儿编程运动:Scratch让每个孩子都能成为创造者(Scratch最新版客户端和初/中/高级学习资料整理分享)

文章目录 &#x1f4d6; 介绍 &#x1f4d6;&#x1f3e1; 演示环境 &#x1f3e1;&#x1f4d2; 文章内容 &#x1f4d2;&#x1f4dd; 安装与使用&#x1f4dd; 社区与资源 &#x1f388; 获取方式 &#x1f388;⚓️ 相关链接 ⚓️ &#x1f4d6; 介绍 &#x1f4d6; 你知道…...

引擎:主程渲染

一、引擎发展 二、引擎使用 1.游戏渲染流程 2.3D场景编辑器操作与快捷键 3.节点的脚本组件 脚本介绍 引擎执行流程 物体节点、声音组件\物理组件\UI组件、脚本组件 暴露变量到面板 4.节点的查找 基本查找 this.node&#xff1a;挂载当前脚本的节点A&#xff1b; this.nod…...

Java 高级面试问题及答案

问题6&#xff1a;请解释Java中的异常处理机制。 探讨过程&#xff1a; 异常处理是Java程序中错误处理的关键部分。正确地处理异常可以提高程序的稳定性和健壮性。 答案&#xff1a; Java中的异常处理机制允许程序在出现错误时&#xff0c;不会导致程序立即终止&#xff0c;而…...

邮件的安全认证(dkim/spf/dmarc)

dkim dkim是用来识别电子邮件合法以及完整性的一种技术手段&#xff0c;主要方式是通过非对称加密对邮件本身进行签名&#xff0c;邮件接收方可以使用发送方提供的公钥对签名进行校验&#xff0c;来确认邮件是否伪造或者被篡改。 如何查看dkim dkim签名被放在邮件原始内容的…...

单调栈问题

原理 单调栈的核心原理是&#xff1a;在栈内保持元素的单调性&#xff08;递增或递减&#xff09; 单调递增栈&#xff1a; 用于处理“下一个更小的元素”问题。当新元素比栈顶元素小或等于时&#xff0c;直接入栈&#xff1b;否则&#xff0c;一直从栈顶弹出元素&#xff0c…...

Hexo博客重新部署与Git配置

由于电脑重装了一次&#xff0c;发现之前Hexo与NexT主题版本过于落后&#xff0c;重新部署了下。 1 Node.js与git安装 这一块安装就不赘述了。去两个官网找安装文件安装即可。 node.js git 打开git以后配置的几个关键命令行。 git config --global user.name "你的gi…...

KUKA机器人专业名词解释

1、CCU Cabinet Control Unit &#xff08;控制柜控制单元&#xff09; 2、CIB Cabinet Interface Board &#xff08;控制柜接口板&#xff09; 3、HMI Human Machine Interface &#xff08;人机界面&#xff09;&#xff1b;KUKA.HMI 是 KUKA 操作界面。 4、KCB …...

阿里云 物联网平台 MQTT连接、数据传输

阿里云 物联网平台 MQTT连接、数据传输 1、设备连接阿里云 2、多设备之前的通信、数据流转 3、设备数据来源的读取。 基于C# winform 开发上位机&#xff0c;读取设备、仪器、MES或者电子元器件的数据&#xff0c;MQTT传输至阿里云平台&#xff0c;可视化界面构建界面&#…...

栈和队列OJ练习题及解答

前言 上一篇博客已经讲到了栈和队列的数据结构&#xff0c;概括一下&#xff1a;栈后进先出&#xff08;Last In First Out&#xff09;、队列先进先出&#xff08;First In First Out&#xff09;。那么&#xff0c;接下来就来讲讲&#xff0c;关于栈和队列的相关练习题&#…...

渗透测试-信息收集

网络安全信息收集是网络安全领域中至关重要的一环&#xff0c;它涉及到对目标系统、网络或应用进行全面而细致的信息搜集和分析。这一过程不仅有助于理解目标网络的结构、配置和潜在的安全风险&#xff0c;还能为后续的渗透测试、风险评估和安全加固提供有力的支持。 在网络安…...

电力乙级资质延伸换证:企业转型的契机

电力乙级资质延伸换证不仅是企业合规运营的必要步骤&#xff0c;同时也为企业转型提供了重要的契机。在这个过程中&#xff0c;企业可以重新审视自身的业务模式、管理体系、技术能力等方面&#xff0c;寻找新的增长点和发展方向。 首先&#xff0c;电力乙级资质延伸换证要求企业…...

基于Redis实现分布式锁——Java版本

基于Redis实现分布式锁——Java版本 版本一版本二版本三Redisson 定义分布式锁接口如下&#xff1a; public interface ILock {boolean tryLock(long timeoutSec);void unlock(); }版本一 设定业务超时时间&#xff0c;到期自动解锁。缺点是超时时间不好估计&#xff0c;需要…...

Qt自定义控件--提升为

为什么要自定义控件 1&#xff0c;有复合小控件需要组合为一个整体控件时&#xff1b; 2&#xff0c;一个复合控件需要重复使用时&#xff1b; 实现 自定义控件文件 新增三个文件 关联不同组的控件 关联之前的准备工作 1&#xff0c;在主控件选择和子控件所有控件所在控件…...

Lua 基础 01 入门

Lua 基础相关知识 第一期 注释 -- 单行注释--[[多行注释 --]]-- 多加一个横杠符号就能重新启用注释内的代码 ---[[print("Lua") --]]数据类型 Lua 是动态类型语言&#xff0c;变量不需要类型定义&#xff0c;只需要为变量赋值。 Lua 有 8 种基本类型&#xff1a…...

RISC-V开发板深度测评指南:从硬件解析到生态实战

1. 项目概述&#xff1a;一次深度参与RISC-V生态的实战机会最近&#xff0c;电子发烧友社区联合多家厂商发起的第二届RISC-V开发板测评大赛&#xff0c;吸引了圈内不少工程师和爱好者的目光。其中&#xff0c;昊芯&#xff08;Haawking&#xff09;作为一家专注于RISC-V处理器I…...

LinuxCNC新手到专家:5个步骤打造你的完美数控系统

LinuxCNC新手到专家&#xff1a;5个步骤打造你的完美数控系统 【免费下载链接】linuxcnc LinuxCNC controls CNC machines. It can drive milling machines, lathes, 3d printers, laser cutters, plasma cutters, robot arms, hexapods, and more. 项目地址: https://gitcod…...

Hi3403开发板内核升级至Linux 6.6:驱动适配与稳定性调优实战

1. 项目概述&#xff1a;一次内核升级背后的工程实践最近&#xff0c;我们团队完成了对迅为iTOP-Hi3403开发板配套SDK的一次重要更新&#xff0c;将内核版本从之前的长期支持版&#xff08;LTS&#xff09;升级到了最新的Linux 6.6。这不仅仅是一个版本号的跳动&#xff0c;对于…...

VS2015安装后找不到控制台项目?别急,你可能只是开错了Blend

VS2015安装后找不到控制台项目&#xff1f;可能是你开错了Blend 刚接触Visual Studio 2015的开发者经常会遇到一个令人困惑的问题&#xff1a;明明安装了VS2015&#xff0c;却找不到Win32控制台应用程序的创建选项。这往往不是因为安装不完整&#xff0c;而是因为误打开了Blend…...

无王无帝定乾坤,来自田间第一人 海棠山铁哥持道定天下

无王无帝定乾坤 ——来自田间第一人千古以来&#xff0c;世人皆认为天下安定、乾坤稳固&#xff0c;必靠帝王集权、朝堂号令、强权治世。 王朝兴替往复&#xff0c;霸业起落无常&#xff0c;靠权柄维系的盛世终难长久&#xff0c;靠杀伐平定的世道终存隐患。 权力会更迭&#x…...

深度解析baidupcsapi:Python百度网盘API高级配置与实战指南

深度解析baidupcsapi&#xff1a;Python百度网盘API高级配置与实战指南 【免费下载链接】baidupcsapi 百度网盘api 项目地址: https://gitcode.com/gh_mirrors/ba/baidupcsapi baidupcsapi是一个功能强大的Python百度网盘API库&#xff0c;为开发者提供了完整的百度网盘…...

从LED闪烁到任务调度:手把手教你用英飞凌AURIX的STM系统定时器构建简单时间片

从LED闪烁到任务调度&#xff1a;基于英飞凌AURIX的STM系统定时器构建轻量级时间片框架 在嵌入式开发中&#xff0c;系统定时器&#xff08;STM&#xff09;常被简化为"高级延时工具"&#xff0c;但它的潜力远不止于此。当开发者面对需要同时处理LED状态控制、按键扫…...

突发!Gemini Ultra最新v1.5更新导致批量推理吞吐下降38%?我们48小时内完成全链路压测并定位CUDA内核缺陷

更多请点击&#xff1a; https://codechina.net 第一章&#xff1a;Gemini Ultra性能测试的背景与挑战 随着多模态大模型能力边界持续拓展&#xff0c;Gemini Ultra作为Google最新发布的旗舰级AI模型&#xff0c;在推理深度、上下文理解与跨模态协同方面提出了前所未有的工程验…...

FPGA资源省一半?揭秘多相滤波器在抽取/内插中的高效实现结构与Xilinx IP核配置要点

FPGA资源优化实战&#xff1a;多相滤波器在采样率转换中的高效架构与Xilinx IP核深度配置 当信号处理系统面临严苛的资源约束时&#xff0c;工程师们常常需要在性能与成本之间走钢丝。多相滤波器结构就像一位精明的财务顾问&#xff0c;能帮你在FPGA资源预算紧张的情况下&#…...

别再只用if-else了!Matlab里switch/case的5个高效用法与避坑指南

别再只用if-else了&#xff01;Matlab里switch/case的5个高效用法与避坑指南 在Matlab编程中&#xff0c;if-else语句几乎是每个开发者最先掌握的控制结构之一。但当你开始处理更复杂的条件逻辑时&#xff0c;一长串的if-elseif-else语句不仅让代码变得难以阅读&#xff0c;还可…...