IPC 进程间通信方式——共享内存
共享内存
- 共享内存区域是被多个进程共享的一部分物理内存。
- 多个进程都可以把共享内存映射到自己的虚拟空间。所有用户空间的进程要操作共享内存,都要将其映射到自己的虚拟空间,通过映射的虚拟内存空间地址去操作共享内存,从而达到进程间的数据通信。
- 共享内存是进程间共享数据的一种最快的方法,一个进程向共享内存区域写入了数据,共享这个内存区域的所有进程就可以立刻看到其中的内容
- 本身不提供同步机制,可通过信号量进行同步
- 提升数据处理效率,一种效率最高的IPC机制

共享内存属性信息
struct shmid_ds{
struct ipc_perm shm_perm;
size_t shm_semsz;//共享内存大小
pid_t shm_lpid;//最后一次调用系统pid
pid_t shm_cpid;//pid的创建者的id
shmatt_t shm_nattch;//和共享内存成功映射的数量
time_t shm_atime;//最后一个成功映射的时间
time_t shm_dtime;//最后断开映射的时间
time_t shm_ctime;//最后改变的时间
}
共享内存使用步骤
- 使用shmget()函数创建共享内存
- 使用shmat()函数映射共享内存,将这段创建的共享内存映射到具体的进程虚拟内存空间
创建共享内存
#include <sys/shm.h>
itn shmget(key_t key,size_t size,int shmflag);
//返回,成功返回内核中共享内存的表示iD,失败返回-1.
- 参数
- key:用户指定的共享内存键值
- size:共享内存大小
- shmflg:IPC_CREAT,IPC_EXCL等权限组合
- erron
- EINVAL(无效的内存段大小)
- EEXIST(内存段已经存在,无法创建)
- EIDRM(内存段已经被删除)
- ENOENT(内存段不存在)
- EACCES(权限不够)
- ENOMEM(没有足够内存创建内存段)
共享内存控制
#include<sys/shm.h>
int shmctl(int shmid,int cmd,struct shmid_ds *buf);
- 参数
- shmid:共享内存ID
- buf:共享内存属性指针
- cmd
- IPC_STAT 获取共享内存段属性
- IPC_SET 设置共享内存段属性
- IPC_RMID 删除共享内存段
- SHM_LOCK 锁定共享内存段页面
- SHM_UNLOCK 解锁锁定
共享内存映射和解除
#include<sys/shm.h>
void shmat(int shmid,char *shmaddr,int shmflg);
int shmdt(char *shmaddr);
//失败返回-1
- 参数
- shmid:共享内存ID
- shmaddr:映射到进程虚拟内存空间的地址,系统自动分配
- shmflg:弱shmaddr为0,shmflag也是0
- SHM_RND
- SHMLBA 地址为2的次方
- SHM_RDONLY 只读方式连接
- errno
- EINVAL 无效的IPC ID值或无效的地址
- ENOMEM 没有足够的内存
- EACCES 权限不够
- 子进程不继承父进程创建的共享内存,大家是共享的,子进程继承父进程映射的地址。
案例:
通过共享内存实现进程间通信
#include <sys/shm.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include "tell.h"
int main(void)
{
//创建共享内存
int shmid;
if((shmid=shmget(IPC_PRIVATE,1023,
IPC_CREAT|IPC_EXCL|0777))<0)
{
perror("shmget error");
exit(1);
}
pid_t pid;
init();//初始化管道
if((pid=fork())<0)
{
perror("fork error");
exit(1);
}else if(pid>0)
{
//parent process
//进行共享内存的映射
int *pi=(int*)shmat(shmid,0,0);
if(pi==(int*)-1)
{
perror("shmat error");
exit(1);
}
//往共享内存中写入数据
*pi=100;
*(pi+1)=200;
//操作完毕,解除映射
shmdt(pi);
//通知子进程去读取数据
notify_pipe();
destory_pipe();
wait(0);
}else
{
//child process
//子进程阻塞,都带父进程往共享内存写入数据
wait_pipe();
//子进程从共享内存中读取数据
//子进程进行共享内存的映射
int *pi=(int*)shmat(shmid,0,0);
if(pi==(int*)-1)
{
perror("shmat error");
exit(1);
}
printf("start:%d,end:%d\n",*pi,*(pi+1));
//读取完毕解除映射
shmdt(pi);
//删除共享内存
shmctl(shmid,IPC_RMID,NULL);
destory_pipe();
}
}
IPC 进程间通信方式——共享内存的更多相关文章
- Linux下进程间通信方式——共享内存
1.什么是共享内存? 共享内存就是允许两个或多个进程共享一定的存储区.就如同 malloc() 函数向不同进程返回了指向同一个物理内存区域的指针.当一个进程改变了这块地址中的内容的时候,其它进程都会察 ...
- Linux进程IPC浅析[进程间通信SystemV共享内存]
Linux进程IPC浅析[进程间通信SystemV共享内存] 共享内存概念,概述 共享内存的相关函数 共享内存概念,概述: 共享内存区域是被多个进程共享的一部分物理内存 多个进程都可把该共享内存映射到 ...
- Linux环境进程间通信(五): 共享内存(下)
linux下进程间通信的几种主要手段: 管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允 ...
- Linux环境进程间通信(五): 共享内存(上)
linux下进程间通信的几种主要手段: 管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允 ...
- 浅析Linux下进程间通信:共享内存
浅析Linux下进程间通信:共享内存 共享内存允许两个或多个进程共享一给定的存储区.因为数据不需要在客户进程和服务器进程之间复制,所以它是最快的一种IPC.使用共享内存要注意的是,多个进程之间对一给定 ...
- Linux进程间通信—使用共享内存
Linux进程间通信-使用共享内存 转自: https://blog.csdn.net/ljianhui/article/details/10253345 下面将讲解进程间通信的另一种方式,使用共享内 ...
- IPC最快的方式----共享内存(shared memory)
在linux进程间通信的方式中,共享内存是一种最快的IPC方式.因此,共享内存用于实现进程间大量的数据传输,共享内存的话,会在内存中单独开辟一段内存空间,这段内存空间有自己特有的数据结构,包括访问权限 ...
- linux进程间通信之共享内存篇
本文是对http://www.cnblogs.com/andtt/articles/2136279.html中共享内存(上)的进一步阐释说说明 1 共享内存的实现原理 共享内存是linux进程间通讯的 ...
- Linux IPC实践(8) --共享内存/内存映射
概述 共享内存区是最快的IPC形式.一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据(如图). 共享内存 VS ...
随机推荐
- Linux后台运行python程序并输出到日志文件
后台运行python程序并标准输出到文件 现在有test.py程序要后台部署, 里面有输出内容 使用命令: nohup python -u test.py > test.log 2>&am ...
- TensorFlow实战第六课(过拟合)
本节讲的是机器学习中出现的过拟合(overfitting)现象,以及解决过拟合的一些方法. 机器学习模型的自负又表现在哪些方面呢. 这里是一些数据. 如果要你画一条线来描述这些数据, 大多数人都会这么 ...
- 【DSP开发】【VS开发】YUV与RGB格式转换
[视频处理]YUV与RGB格式转换 YUV格式具有亮度信息和色彩信息分离的特点,但大多数图像处理操作都是基于RGB格式. 因此当要对图像进行后期处理显示时,需要把YUV格式转换成RGB格式. RGB与 ...
- Elasticsearch-字符串类型
ES-用于定义文档字段的核心类型 ES中一个字段可以是核心类型之一,如字符串.数值.日期.布尔型,也可以是一个从核心类型派生的复杂类型,如数组. 字符串类型 索引一类型为字符串的数据doc1: Fen ...
- Websocket 突破最大长连接
为了测试机器能够最大的长连接个数,故写了一个js脚本,需要用node进行执行 var WebSocketClient = require('websocket').client; var size = ...
- 使用vs工具查看dll依赖(也可查看pyc文件的依赖)
vs工具中有个工具叫dumpin.exe,可以用来查看exe文件.dll文件.pyc文件依赖于哪些dll,从而针对性地去检查具体缺失哪些文件(目前是在装TensorFlow时查看具体需要哪个版本的cu ...
- Volatile可见性 与 Synchronization原子性的优化
Volatile可见性 比如现在我们有这样一段代码:线程等待另一个线程将数据装载完就输出success,可是最后程序一直卡在while循环里没有往下执行. public class VolatileD ...
- 如何给Word参考文献加方括号
脚注是在页面下端添加的注释,如添加在一篇论文首页下端的作者情况简介:尾注是在文档尾部(或节的尾部)添加的注释,如添加在一篇论文末尾的参考文献目录. 若用手工添加脚注和尾注,则既麻烦又易出错.当用Wor ...
- 099、如何访问Service (Swarm06)
参考https://www.cnblogs.com/CloudMan6/p/7909136.html 前面已经学习了如何部署Service吗,也验证了swarm的failover特性,下面我们要学 ...
- MyBatis与Hibernate总结篇
也用了这么久的Hibernate和MyBatis了,一直打算做一个总结,就他们之间的优缺点说说我自己的理解: 首先,Hibernate是一个ORM的持久层框架,它使用对象和我们的数据库建立关系,在Hi ...