1.特点:
  1)共享内存是一种最为高效的进程间通信方式,进程可以直接读写内存,而不需要任何数据的拷贝。如管道当在内核空间创建以后,用户空间需要内存  拷贝,需要拷贝数据,所以效率低。

  2)为了在多个进程间交换信息,内核专门留出了一块内存区,可以由需要访问的进程将其映射到自己的私有地址空间

  3)进程就可以直接读写这一内存区而不需要进行数据的拷贝,从而大大提高的效率。

  4)由于多个进程共享一段内存,因此也需要依靠某种同步机制,如互斥锁和信号量等

2.共享内存的使用步骤:

  1)创建/打开共享内存

  2)映射共享内存。即把指定的共享内存映射到进程的地址空间用于访问

  3)撤销共享内存映射

  4)删除共享内存对象

3.相关函数:

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

    功能:产生一个独一无二的key值

    参数:Pathname:已经存在的且可访问文件的名字

       Proj_id:一个字符(因为只用低8位),生成key的数字,不能为0

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

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

    功能:创建或打开共享内存对象

    参数:key 键值 ; size 共享内存的大小 ; shmflg IPC_CREAT|IPC_EXCL|0777(标志位,是否新建,和权限)

    返回值:成功 shmid(共享内存ID) ; 出错 -1

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

    功能:映射共享内存,即把指定的共享内存映射到进程的地址空间用于访问

    参数:shmid 共享内存的id号 ; shmaddr 一般为NULL,表示由系统自动完成映射,如果不为NULL,那么有用户指定

             shmflg:SHM_RDONLY就是当前进程对该共享内存只能进行读操作  0 可读可写

    返回值:成功:完成映射后的地址,出错:-1的地址

    用法:(p = (char *)shmat(shmid,NULL,0)) == (char *)-1  (对-1强转成字符型地址)

  4)int shmdt(const void *shmaddr);

    功能:取消映射

    参数:要取消的地址

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

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

    功能:(删除共享内存),对共享内存进行各种操作

    参数:shmid 共享内存的id号 ; cmd IPC_STAT 获得shmid属性信息,存放在第三参数

          IPC_SET 设置shmid属性信息,要设置的属性放在第三参数

          IPC_RMID:删除共享内存,此时第三个参数为NULL即可

    返回: 成功0 ; 失败-1

    用法:shmctl(shmid,IPC_RMID,NULL);

补充:

1、IPC对象创建后一直存在,直到被显示的删除,及最后一个进程要删除IPC对象

2、每个IPC对象除了有ID之外还有相关联的key值(IPC对象的属性),key值作用,通过key值,不同的进程能够找到打开同一个IPC对象,因为IPC对象创建好以后,他的ID是随机分配的,只有创建IPC对象的进程才能直接获得这个ID,其他进程是不知道这个ID的,其他进程就通过key去获得这个ID,通过key打开同一个IPC对象。key值为0表示私有的。不同的进程用ftok函数,生成的key值要一样,及ftok函数的参数就要保持一致。

例子:

通过终端输入信息,写入到共享内存中,然后打印共享内存中的数据

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <errno.h> int main(int argc, const char *argv[])
{
key_t key; //共享内存的唯一标识 key 值
int shmid; //
char *p = NULL; // 函数原型 key_t ftok(const char *pathname, int proj_id);;
key = ftok("./app",'a'); //创建key值
if(key < )
{
perror("fail ftok ");
exit();
}//共享内存 不存在,创建
shmid = shmget(key,,IPC_CREAT|IPC_EXCL|);// 创建/打开共享内存,返回id根据id映射
if(shmid < )
{
if(errno == EEXIST)//文件存在时,直接打开文件获取shmid
{
printf("file eexist");
shmid = shmget(key,,); //如果创建了,直接打开
}
else
{
perror("shmget fail ");
exit();
}
}
p = (char *)shmat(shmid,NULL,);//映射,返回地址,根据地址操作,错误 返回 -1 的地址
if( p == (char *)(-) ) //错误形式判断
{
perror("shmat fail ");
exit();
} read(,p,);//从终端读数据,写入p指向的空间
printf("%s\n",p); //打印p指向空间的内容 shmdt(p);//解除映射 //int shmctl(int shmid, int cmd, struct shmid_ds *buf);
shmctl(shmid,IPC_RMID,NULL); //删除 return ;
}

测试:终端输入123 写入共享内存中,然后通过共享内存打印出来

共享内存总结;

(1)key 值得获取,根据不同的key值获得这块共享内存唯一的id 根据这个唯一的 id 不同进程实现共同访问这块共享内存(进程间想访问同一块共享内存,就要得到这个共享内存唯一的id)

(2) key值通过函数 ftok 函数获取,函数的功能是根据不同的文件名,不同的字符得到这个不同的 key 值

(3) 当然这个key值其实就是一个整数,也可以通过给定确定的数创建 / 打开一块共享内存,使用 ftok 的目的是防止自己乱填数字,造成不同进程间访问冲突

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

  1. Linux进程间通信—共享内存

    五.共享内存(shared memory) 共享内存映射为一段可以被其他进程访问的内存.该共享内存由一个进程所创建,然后其他进程可以挂载到该共享内存中.共享内存是最快的IPC机制,但由于linux本身 ...

  2. linux 进程间通信 共享内存 mmap

    共享内存可以说是最有用的进程间通信方式,也是最快的IPC形式.两个不同进程A.B共享内存的意思是,同一块物理内存被映射到进程A.B各自的进程地址空间.进程A可以即时看到进程B对共享内存中数据的更新,反 ...

  3. linux进程间通信-共享内存

    转载:http://www.cnblogs.com/fangshenghui/p/4039720.html 一 共享内存介绍 共享内存可以从字面上去理解,就把一片逻辑内存共享出来,让不同的进程去访问它 ...

  4. Linux进程间通信 共享内存+信号量+简单样例

    每个进程都有着自己独立的地址空间,比方程序之前申请了一块内存.当调用fork函数之后.父进程和子进程所使用的是不同的内存. 因此进程间的通信,不像线程间通信那么简单.可是共享内存编程接口能够让一个进程 ...

  5. linux 进程间通信 共享内存 shmat

    系统调用mmap()通过映射一个普通文件实现共享内存.系统V则是通过映射特殊文件系统shm中的文件实现进程间的共享内存通信.也就是说,每个共享内存区域对应特殊文件系统shm中的一个文件(这是通过shm ...

  6. Linux环境进程间通信: 共享内存

    Linux环境进程间通信: 共享内存 第一部分 共享内存可以说是最有用的进程间通信方式,也是最快的IPC形式.两个不同进程A.B共享内存的意思是,同一块物理内存被映射到进程A.B各自的进程地址空间.进 ...

  7. Linux IPC 共享内存

    共享内存 共享内存(shared memory)是最简单的Linux进程间通信方式之一. 使用共享内存,不同进程可以对同一块内存进行读写. 由于所有进程对共享内存的访问就和访问自己的内存空间一样,而不 ...

  8. C# 进程间通信(共享内存)

    原文:C# 进程间通信(共享内存) 进程间通信的方式有很多,常用的方式有: 1.共享内存(内存映射文件,共享内存DLL). 2.命名管道和匿名管道. 3.发送消息 本文是记录共享内存的方式进行进程间通 ...

  9. Linux下进程间通信--共享内存:最快的进程间通信方式

    共享内存: 一.概念: 共享内存可以说是最有用的进程间通信方式,也是最快的IPC形式.两个不同进程A.B共享内存的意思是,同一块物理内存被映射到进程A.B各自的进程地址空间. 进程A可以即时看到进程B ...

随机推荐

  1. Javascript中的相等比较

    在比较相等或不相等之前,会对操作数进行类型转换,然后比较相等性 在转换不同的数据类型时,相等和不相等操作符遵循下列基本规则: 1.如果由一个操作数是布尔值,则在比较相等性之前先将其转换为数值:2.如果 ...

  2. 防止按钮重复点击的思路(js篇)

    最直接的思路可能就是点击按钮后,按钮的事件绑定函数解绑,1s后重新绑定函数 <button id=</button> <script> btn.onclick = fun ...

  3. esxi主机用命令行强行关闭通过前端界面无法关闭的ESXI虚拟机

    环境:esxi5.1-esxi6.5 背景:如果esxi下面某一台vm死机了,并且esxi的控制台卡死不能用,为了不影响同一个esx下其他的vm正常使用,那么我们只能用命令行来单独重启此vm,保证一定 ...

  4. java nio socket实例

    Server端代码: public class NioServer { //通道管理器 private Selector selector; //获取一个ServerSocket通道,并初始化通道 p ...

  5. VirtualBox安装CentOS系统

    1. 准备材料 虚拟机软件: VirtualBox 系统iso版本:CentOS-7-x86_64-DVD-1611.iso 虚拟机软件下载地址: https://www.virtualbox.org ...

  6. 第九章 Service

    2019-09-23 今天距离2020年刚好有一百天,希望在未来的百日里能不负期待 不忘初心,方得始终, 初心易得,始终难守. 一.Service 的概念 Kubernetes Service定义了这 ...

  7. zabbix快速安装(Ubuntu18.04, Nginx)

    ubuntu18.04快速安装zabbix4.0 https://blog.csdn.net/qq_33317586/article/details/83867756 需要安装的东西:nginx,ph ...

  8. nodejs 模板引擎jade的简单使用(2)

    1.jade html head style body div.box div#div1 div aaa div(class="aaa left-warp active") div ...

  9. SpringBoot 2.x 使用 JWT(JSON Web Token)

    一.跨域认证遇到的问题 由于多终端的出现,很多的站点通过 web api restful 的形式对外提供服务,采用了前后端分离模式进行开发,因而在身份验证的方式上可能与传统的基于 cookie 的 S ...

  10. php封装支付

    <?php namespace Pay; use think\Config; vendor ('Pay.alipay.pagepay.service.AlipayTradeService'); ...