Linux多进程间通信——共享内存实现

2023-05-16

Linux多进程间通信——共享内存实现

又到了每周分享时刻,这周我要分享的是关于Linux中进程间通信问题,这对于底层程序的实现至关重要,进程间通信方式主要包括管程、共享内存、消息传递、套接字这几种方式,这个专题我主要介绍两种进程间通信方式,共享内存和套接字通信。为什么介绍这两种呢,因为共享内存是实现进程间通信最有效的方式,效率高,创建一块虚拟共享内存之后,所有的进程只要映射到这块内存之后就可以完成通信,但是需要做好进程间的同步与互斥。而socket套接字通信又是实现不同主机进程通信的方式,掌握了这两种通信方式,你就能够实现上位机和开发板之间的通信传输,以及实现开发板中不同进程之间的协作完成一个功能。

下面正式开始我们这个专题的讲解,首先介绍使用共享内存来实现进程间通信的实现。

1、共享内存的实现原理

共享内存顾名思义就是有一块内存是各进程可以共享的,在操作系统也被称为临界区,对临界区的访问需要进行进程间的互斥操作,不然得到的结果就不是所期望的结果。

那共享内存由进程中的哪一个进程创建呢,进程中的哪一个进程都可以创建共享内存,其他进程只需要将共享内存映射到自己的地址空间中就可以访问。

如果我创建了两个共享内存,两个进程间怎么来区分我访问的是同一个共享内存呢,这就需要给共享内存创建一个自己的身份ID,需要访问该共享内存的进程在映射时也带上这个ID,就可以区分访问的共享内存是哪一个了。

下面我们就可以给出创建共享内存的步骤了:

一、创建一个共享内存的键值,也就是给共享内存一个身份证

key_t shmkey = ftok(".", 0);//这里的"."就是身份证

二、用共享内存的键值创建共享内存

shmid = shmget(shmkey, 1024, 0666|IPC_CREAT);

这里的1024表示创建的共享内存的大小,IPC_CREAT表示的时创建共享内存。shmid表示共享内存的标识符。

三、将共享内存标识符映射到本进程的地址空间中

addr = shmdt(shmid, 0, 0);

映射成功之后,我们就可以对addr访问,就相当于访问了共享内存中的数据。

四、删除进程中的共享内存

shmdt(addr);

五、销毁创建的共享内存

shmctl(shmid, IPC_RMID, NULL);

上面五步就是创建共享内存的步骤,其中第五步在互相通信的任一进程中操作即可。

掌握了共享内存的实现原理,下面我们就可以用共享内存来实现进程间的通信了。

2、进程间通信案例

这里我们完成一个写进程和读进程,shmmutexWrite.c向共享内存中写入数据,shmumutexRead.c从共享内存中读取数据。

我先给出shmmutexWrite.c的源代码:

//向共享内存写入数据
#include <semaphore.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <sys/shm.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>

#define BUFFER_SIZE 10
#define sem_name "mysem"

int main(){
	struct People{
		char name[10];
		int age;
	};
	
	int shmid;//定义一个共享内存标识符
	sem_t *sem;//定义一个信号量
	int age = 10, i = 1;
	char buff[BUFFER_SIZE];
	key_t shmkey;//定义一个共享内存的键值
	shmkey = ftok("shmmutexRead.c", 0);
	//创建共享内存和信号量的IPC
	sem = sem_open(sem_name, O_CREAT, 0644, 1);
	if(sem == SEM_FAILED){
		printf("ubable to creat semaphore!\n");
		sem_unlink(sem_name);
		exit(-1);
	}
	shmid = shmget(shmkey, 1024, 0666|IPC_CREAT);
	if(shmid == -1){
		printf("shmget failed!\n");
	}
	struct People *addr;
	//将共享内存映射到当前进程中
	addr = (struct People*)shmat(shmid, 0, 0);
	if(addr == (struct People*)-1){
		printf("shmat failed\n");
	}
	//向共享内存写入数据
	addr->age = 0;
	printf("写进程映射的共享内存地址 = %p\n", addr);
	do{
		sem_wait(sem);//信号量初始值为1,现在减1,为0
		memset(buff, 0, BUFFER_SIZE);
		memset((addr+i)->name, 0, BUFFER_SIZE);
		printf("写进程:输入一些姓名(不超过10个字符)到共享内存:\n");
		if(fgets(buff, BUFFER_SIZE, stdin) == NULL){
			perror("fgets\n");
			sem_post(sem);
			break;
		}
		strncpy((addr + i)->name, buff, strlen(buff) - 1);
		(addr + i)->age = ++age;
		addr->age++;
		i++;
		sem_post(sem);
		sleep(1);
	}while(strncmp(buff, "quit", 4) != 0);
	if(shmdt(addr) == -1){
		printf("shmdt if failed\n");
	}
	sem_close(sem);
	sem_unlink(sem_name);
}

本来shmmutexRead.c的代码想让你们自己写的,练练手,因为两个程序基本一样,下面我也直接给出吧。

//从共享内存中读取数据
#include <semaphore.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <sys/shm.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>

#define sem_name "mysem"

int main(){
	int shmid;
	sem_t *sem;
	int i = 1;
	key_t shmkey;
	shmkey = ftok("shmmutexRead.c", 0);
	struct People{
		char name[10];
		int age;
	};
	
	sem = sem_open(sem_name, 0, 0644, 0);
	if(sem == SEM_FAILED){
		printf("unable open semapore\n");
		sem_close(sem);
		exit(-1);
	}
	shmid = shmget(shmkey, 0, 0666);
	if(shmid == -1){
		printf("shmget failed\n");
		exit(0);
	}
	
	struct People* addr;
	addr = (struct People*)shmat(shmid, 0, 0);
	if(addr == (struct People*)-1){
		printf("shm shmat is failed\n");
		exit(0);
	}
	printf("读进程映射的共享内存地址 = %p\n", addr);
	
	do{
		sem_wait(sem);
		if(addr->age > 0){//说明共享内存已经写入数据
			printf("读进程:绑定到共享内存 %p : 姓名 %d %s, 年龄%d \n", addr, i, (addr + i)->name, (addr+i)->age);
			addr->age--;
			if(strncmp((addr+i)->name, "quit", 4) == 0) break;
			i++;
		}
		sem_post(sem);
	}while(1);
	sem_close(sem);
	
	if(shmdt(addr) == -1) printf("shmdt failed\n");
	if(shmctl(shmid, IPC_RMID, NULL) == -1) printf("shmctl delete error\n");
}

先运行创建共享内存的进程shmmutexWrite.c,在运行shmmutexRead.c。

请添加图片描述

更多嵌入式相关内容,请关注公众号河边小乌龟爬。有什么问题可以在下方留言,我看到之后会回复你。

我是河边小乌龟爬,学习嵌入式软件开发路上的一名小学生,欢迎大家相互交流哇。公众号:河边小乌龟爬。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Linux多进程间通信——共享内存实现 的相关文章

  • org.htmlparser小结

    org htmlparser 主要用来解析HTML网页 一 基本上HTML中的每个标签对应于一个类 xff0c 例如 xff1a p标签对应于ParagraphTag类 ul标签对应于BulletList类 li标签对应于Bullet类 a
  • STM32中使用printf打印串口数据

    STM32使用printf打印串口 摘要 该方法适用于 STM32 xff0c 实现了使用 printf 等标准 C 流函数输出数据的办法 xff0c 极大的减少了输出串口数据时所需要做的数据处理 实现原理 在 C 库中 xff0c pri
  • 尺取法(图文解析、初学推荐)

    文章目录 最少连续页 xff08 poj3320 xff09 分析题意第一步 xff1a 找第一个满足条件区间第二步 xff1a 开始将左端边界向右移 xff0c 达到 缩小 区间 减少连续页的目的 归纳总结代码 尺取法 xff1a 顾名思
  • 第九届蓝桥杯——倍数问题(数论、枚举)

    倍数问题 众所周知 xff0c 小葱同学擅长计算 xff0c 尤其擅长计算一个数是否是另外一个数的倍数 但小葱 只擅长两个数的情况 xff0c 当有很多个数之后就会比较苦恼 现在小葱给了你 n 个数 xff0c 希望你 从这 n 个数中找到
  • 基于Python OpenCV的车牌识别(一)————车牌定位

    一开始本想基于OpenCV原理编写定位程序的 xff0c 但发现效果并不是特别理想 xff0c 只能识别我处理的那张图片的车牌 xff0c 在使用其他图片识别的时候失误非常多 xff0c 几乎识别不到车牌 xff0c 下面附上识别的代码和效
  • 计算机网络——尝试SMTP交互(邮件发送)

    前言 本文通过 Windows 命令提示符 xff08 cmd xff09 实现邮件发送 第一步 xff1a 打开 Windows Telnet Client 功能 打开控制面板 点击程序 点击启动或关闭 Windows 功能 将 Teln
  • 离散数学:推理规则

    推理规则解释说明前提引入规则可以在证明的任何时候引入前提结论引入规则在证明的任何时候 xff0c 已证明的结论都可以作为后续证明的前提置换规则在证明的任何时候 xff0c 命题公式中的任何子命题公式都可以用与之等价的命题公式置换假言推理规则
  • 离散数学:谓词逻辑命题符号化

    个体词 个体常项或个体常元 xff1a 使用 x y z 表示个体变项或个体变元 xff1a 使用 a b c 表示个体域或论域 xff1a 个体变元的取值全总个体域 xff1a 宇宙件一切事物 谓词 概念 xff1a 表示个体性质或彼此之
  • 离散数学:主范式(主析取范式、主合取范式)

    求解主析取范式 主合取范式方法 1 真值表法 在表中列出变元值的全部可能 查表判断命题 命题结果真 xff0c 变元值对应主析取范式 命题结果假 xff0c 变元值对应主合取范式 2 等值演算法 命题化简 蕴涵等值式 xff1a A B A
  • Camera:双摄基本原理

    基本概念 基线 xff1a 两镜头的间距 景深 xff1a 画面清晰范围 xff0c 景深小 浅 背景虚化 xff0c 景深大背景清晰 影响景深的三个因素 xff1a 光圈 xff08 反比 xff0c 光圈越大景深越小 xff09 焦距
  • Linux:Hello World 模块

    前言 Linux 系统为应用程序提供了功能强大且容易扩展的 API xff0c 但在某些情况下 xff0c 这还远远不够 与硬件交互或进行需要访问系统中特权信息的操作时 xff0c 就需要一个内核模块 下面从 Hello World 模块来
  • 基础算法题 —— 计算字符串的编辑距离(动态规划)

    题解 在题目所求的编辑距离是指字符串 a 转换成字符串 b 所需最小的操作步骤 a i 61 b j 时 xff0c 要使 a i 61 61 b j 可使用的操作步骤如下 xff1a a i 可替换成 b j 字符串 a 下标为 i 的位
  • Matlab —— 02 基本操作与矩阵输入

    一 基本操作 1 help 功能 xff1a 便于快速了解指令的使用 ex 查询 pi 的简明介绍 Question 通过 matlab 计算出下列算式结果 Answer 1 2 或 xff08 3 xff09 xff08 4 xff09
  • Matlab —— 03结构化程式与自定义函数

    Script writing 新建脚本编辑界面 编辑测试文档 运行节 将整个框架划分为多个小节运行 xff0c 便于进行 Debug 设置断点 检查变量数值 方法一 xff1a 工作区内检查 方法二 xff1a 鼠标滑到变量位置 Struc
  • [unity] NavMesh 寻路 贴边走的问题 优化。

    NavMesh 寻路出来后 xff0c 角色走路很多时候贴着地图的边缘走 xff0c 很别扭 网上找了一个类似的问题 xff1a http www manew com thread 94163 1 1 html 里面方案 也挺适合我们的情况
  • 网易云ncm文件转mp3

    本人卖了一个ipod nano6 xff0c 帮买家导些歌 xff0c 结果用网易云下下来一堆ncm文件 xff0c 一脸懵逼 xff0c 因为ipod只能放mp3文件 上网查了一下ncm文件 xff0c 只能在会员有效期内放这个歌 于是上
  • catkin_make编译时fatal error: ###_msgs/***.h: 没有那个文件或目录

    h是头文件 xff0c 头文件 xff0c 就是一些定义和声明 xff0c 注意是定义 xff08 算是API接口吧 xff09 xff0c 编译时先加载这些头文件 xff0c 加载好了后主程序才能调用这些定义 那出现这种错误就是编译该程序
  • Theta* : 基于网格的任意角度寻路

    原文地址 xff08 科学上网 xff09 xff1a https arxiv org ftp arxiv papers 1401 1401 3843 pdf 1 简介 在本文中 xff0c 我们将研究一种机器人技术或视频游戏中使用的路径规
  • deepsort之YOLO系列目标跟踪及其他功能

    文章目录 一 ID switch是什么 xff1f 二 deepsort框架二 deepsort需要的模型1 目标检测器2 目标跟踪器 一 ID switch是什么 xff1f 跟踪问题中最重要的就是数据关联 xff08 data asso
  • PX4软件在环仿真注意点

    注 xff1a 最新内容参考PX4 user guide 点击此处 PX4下载指定版本代码和刷固件的三种方式 点击此处 PX4sitl固件编译方法 点击此处 PX4开发指南 点击此处 PX4无人机仿真 Gazebo 点击此处 px4仿真 知

随机推荐

  • cmake:生成动态链接库并使用

    已知 库函数的源文件名字叫hello c xff0c 库函数的头文件名字叫hello h xff0c 生成的动态链接库函数的名称叫libhello so xff0c 目标是在 hello install文件夹下安装这个库 xff0c 以便后
  • 算法:皇后问题

    问题 国际象棋中的皇后 xff0c 可以横向 纵向 斜向移动 如何在一个NXN的棋盘上放置N个皇后 xff0c 使得任意两个皇后都不在同一条横线 竖线 斜线方向上 xff1f 举个栗子 xff0c 下图的绿色格子是一个皇后在棋盘上的 封锁范
  • OpenCV+Python二维码条形码识别

    先上源码 xff0c github地址 xff1a https github com DerrickRose25 Opencv QRcode recognition 环境 xff1a Pycharm Python3 7 在pycharm里安
  • Onvif协议:实现Probe命令来进行设备发现(discover)

    在onvif协议对接中 xff0c 首先要明确服务器和客户端的身份 服务器 xff1a 通常是你要对接的其他厂家的数字摄像头 xff08 IPC xff09 客户端 xff1a 通常是对接的ipc的设备程序 xff0c 安防业内多称 xff
  • Linux C/C++编程:Udp组播(多播)

    Udp多播简介 概叙 单播用于两个主机之间单对单的通信广播用于一个主机对整个局域网上所有主机上的数据通信单播和广播是两个极端 xff0c 要么对一个主机进行通信 xff0c 要么对整个局域网的主机进行通信实际情况下 xff0c 经常需要对一
  • cmake:同一目录下多个源文件

    此文为 xff1a 轻松入门cmake系列教程前文为 xff1a cmake xff1a Hello cmake 接下来进入稍微复杂的例子 xff1a 在同一个目录下有多个源文件 第一个实验 实践 在之前的目录下添加2个文件 xff0c t
  • cmake:string

    字符串操作 概要 Search span class token operator and span Replace span class token function string span span class token punctu
  • 性能:你知道并发用户数应该怎么算吗

    我们知道 xff0c 一个性能测试中 xff0c 往往会有各种各样的指标 xff0c 比如TPS RPS QPS HPS CPM等 我们在实际工作的时候 xff0c 应该对这些概念有统一的认识 建议使用TPS作为关键的性能指标 另外 xff
  • C/C++面试:手写智能指针类

    shared ptr原理 shared ptr实际上是对裸指针进行了一层封装 xff0c 成员变量除了裸指针之外 xff0c 还有一个引用计数 xff0c 它记录裸指针被引用的次数 xff08 有多少个shared ptr指向这同一个裸指针
  • golang:http.request

    request 表示由服务器接收或由客户端发送的HTTP请求 xff0c 例如客户端 client 在发送各种请求时 xff0c 需要先新建一个请求对象 xff0c 然后调用一些请求的方法开始自定义一些配置 xff0c 服务端监听到该请求便
  • git:smartgit

    下载安装smartgit 下载 xff1a https www syntevo com smartgit download 终端下操作 xff1a 执行命令 xff1a tar xvf smartgit tar gz 执行命令 xff1a
  • golang:如何在proto3中用上golang对应的interface{}类型

    首先 xff0c 我希望所有golang中用于http请求响应的结构 xff0c 都使用proto3来定义 麻烦的是 xff0c 有的情况下某个字段的类型可能是动态的 xff0c 对应的JSON类型可能是number string bool
  • 如何计算icmp校验和

    前几天看到大佬写的一篇关于icmp远控后门文章 xff0c 对icmp协议充满了激情 xff0c 通过查阅资料了解相关所需的知识 xff0c 实现整个程序首先要了解imcp包请求与回复 xff0c 在整个请求中最先就涉及到icmp包的构造
  • 作为一个自动化本科生到底应该学些什么(讲讲个人经历和感受)

    2019 6 26日深夜 晚上在床上准备休息 xff0c 一个同跟我聊天说自己大学荒废了 xff0c 突然有很多感想想写一写 我觉得自己大学也是得过且过地过着的 xff0c 不过虽然充满艰辛和不满意 xff0c 但还是挺充实的 先自我介绍一
  • 嵌入式项目实战——基于QT的视频监控系统设计(一)

    嵌入式项目实战 基于QT的视频监控系统设计 xff08 一 xff09 这个五一因为疫情 xff0c 只能待在家里 xff0c 想了想不如将我之前做的一个小的嵌入式的练习项目分享出来 xff0c 供入门嵌入式的同学们学习 基于QT的视频监控
  • 嵌入式项目实战——基于QT的视频监控系统设计(三)

    嵌入式项目实战 基于QT的视频监控系统设计 xff08 三 xff09 进入到五一假期第三天 xff0c 继续我们的项目 本来五一假期还是想好好休息一下的 xff0c 因为最近学习的状态不太好 xff0c 刷题都没有思路了 xff0c 但是
  • 嵌入式开发板RS485协议串口编程——角度传感器数据读取

    嵌入式开发板RS485协议串口编程 倾角传感器数据读取 之前分享过一篇嵌入式操作系统开发板中的串口编程 光敏电阻数据读取 xff0c 是基于TTL协议的串口编程 xff0c 本节主要讲述基于RS485协议的串口编程 xff0c 掌握了这两种
  • 嵌入式开发板CAN通信编程——伺服电机驱动

    嵌入式开发板CAN通信编程 伺服电机驱动 在实际的嵌入式项目开发过程中 xff0c 若不涉及上位机与开发板的通信传输数据 xff0c 那最关键的无非就是两个内容 xff0c 读取传感器的数据并处理 xff0c 驱动硬件设备工作 传感器数据的
  • 嵌入式字符设备驱动——ULN2003步进电机驱动程序实现

    嵌入式字符设备驱动 ULN2003步进电机驱动程序实现 之前分享了字符设备驱动程序的实现 hello驱动 xff0c 是不涉及硬件操作的 xff0c 我说过要给大家分享一篇涉及硬件操作的字符设备驱动程序的实现 xff0c 今天周末休息 xf
  • Linux多进程间通信——共享内存实现

    Linux多进程间通信 共享内存实现 又到了每周分享时刻 xff0c 这周我要分享的是关于Linux中进程间通信问题 xff0c 这对于底层程序的实现至关重要 xff0c 进程间通信方式主要包括管程 共享内存 消息传递 套接字这几种方式 x