1. 共享内存:

共享内存方式可以在多个进程直接共享数据,因为其直接使用内存,不要多余的拷贝,是速度最快的IPC方式;

共享内存有两种实现方式,使用mmap和shm方式,如下图:

(1) mmap方式是将文件与进程地址空间进行映射,对实际物理内存影响小;

(2) shm方式是将每个进程的共享内存与实际物理存储器进行映射,对实际物理内存影响大;

由于XSI IPC自身缺点,所以建议使用mmap来实现共享内存;

2. mmap相关函数原型:

#include <sys/mman.h>

void *mmap(void *addr, size_t len, int prot, int flag, int filedes, off_t off);

ret = 成功返回映射区域开始地址,失败返回MAP_FAILED

addr-参数用来指定映射区域的起始地址。通常设置为0,表示有系统选择映射区域的起始位置;

filedes-要映射文件的描述符。在映射文件到地址空间之前,先要打开文件;

len-映射的字节数;

off-要映射字节在文件中的起始偏移量;

prot-对映射存储区的保护要求,指定为PROT_NONE(不可访问), PROT_READ(可读), PROT_WRITE(可写), PROT_EXEC(可执行)的任意组合或者按位或;

但是对于映射存储区的保护要求不能超过文件open的权限,比如文件是只读打开的,则不能指定PROT_WRITE;

flag-MAP_FIXED 返回值必须等于addr,如果未制定此标志,addr非0,则作为一种建议,内核不保证会使用该地址。为了获得最大的可移植性,建议addr指定为0;

MAP_SHARED 说明了本进程对映射区域进行储存操作的配置。此标志指定存储操作修改映射文件,也就是相当于对该文件执行write;

MAP_PRIVATE 本标志说明对映射区存储操作会创建该映射的一个副本。所以后续操作都会引用该副本,而不是原始文件;

MAP_SAHRED 和 MAP_PRIVATE 必须指定其一,但是不能同时指定;

修改现有映射区的权限,与上面mmap中的prot字段相同;

#include <sys/mman.h>

int mprotect(void *addr, size_t len, int prot);

ret = 成功返回0,失败返回-

如果共享内存中的页已被修改,则可以调用msync将该页冲洗到映射的文件中;

如果映射是私有的,那么不修改映射文件;

#include <sys/mman.h>

int msync(void *addr, size_t len, int flags);

ret = 成功返回0 失败返回-

flags- MS_ASYNC和MS_SYNC必选其一;

   MS_ASYNC-异步sync,无需等待,系统调度;

MS_SYNC-同步sync,需要等到操作完成;

MS_INVALIDATE-可选标志,通知操作系统丢弃没有同步的页;--一般不使用;

进程终止,或者调用munmap,内存映射就被自动解除,关闭文件描述符filedes并不会解除映射;

调用munmap不会是映射内存内容写到磁盘文件,share类型写磁盘是内核根据算法自动执行,private则丢弃修改;

#include <sys/mman.h>

int munmap(caddr_t addr, size_t len);

ret-成功返回0 失败返回-

3.  文件映射到进程位置:

文件映射到进程地址空间中堆和栈直接的一段内存。

4. 子进程影响:

fork之后,因为子进程复制父进程的地址空间,而存储映射是该地址空间的一部分,所以子进程继承了该存储映射区域,但是当调用了exec函数组之后的新程序则不继承该映射区;

测试代码:--测试两个进程通过mmap映射信息,测试共享时,数据情况,和共享后数据情况;

common.h

 #include <sys/mman.h>
#include <stdlib.h>
#include <stdio.h>
#include <fcntl.h>
#include <string.h> #define MMAP_FILE "/var/tmp/test_mmap"
#define NAME_LEN 64 typedef struct book {
int id;
char name[NAME_LEN];
}book_t;

mmap_proc1.c -- 进程1,做映射,映射超过文件长度的区域

 #include "common.h"

 int main(int argc, char *argv[])
{
int fd = -;
int i = ;
book_t *book = NULL; if ((fd = open(MMAP_FILE, O_RDWR | O_CREAT | O_TRUNC)) < ){
perror("open file error\n");
return -;
} if (lseek(fd, sizeof(book_t) * - , SEEK_SET) < ){
perror("lseek error\n");
close(fd);
return -;
} write(fd, "", ); if ((book = (book_t *)mmap(NULL, sizeof(book_t) * , PROT_READ|PROT_WRITE, MAP_SHARED, fd, )) == MAP_FAILED){
perror("mmap error\n");
return -;
} close(fd); for (i = ; i < ; i++){
char name[] = { };
snprintf (name, , "book-%d", i); (*(book + i)).id = i;
strncpy((*(book + i)).name, name, NAME_LEN - );
} sleep(); munmap(book, sizeof(book_t) * ); return ;
}

mmap_proc2.c -- 进程2,做映射,映射与进程1相同区域

 #include "common.h"

 int main(int argc, char *argv[])
{
int fd = -;
int i = ;
book_t *book = NULL; if ((fd = open(MMAP_FILE, O_RDWR | O_CREAT)) < ){
perror("open file error\n");
return -;
} if ((book = (book_t *)mmap(NULL, sizeof(book_t) * , PROT_READ|PROT_WRITE, MAP_SHARED, fd, )) == MAP_FAILED){
perror("mmap error\n");
return -;
} close(fd); for (i = ; i < ; i++){
printf("id:%d, name:%s\n", (*(book+i)).id, (*(book+i)).name);
} munmap(book, sizeof(book_t) * ); return ;
}

Linux-进程间通信(三): 共享内存的更多相关文章

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

    Linux进程间通信-使用共享内存 转自: https://blog.csdn.net/ljianhui/article/details/10253345 下面将讲解进程间通信的另一种方式,使用共享内 ...

  2. Linux进程间通信——使用共享内存

    一.什么是共享内存 顾名思义,共享内存就是允许两个不相关的进程访问同一个逻辑内存.共享内存是在两个正在运行的进程之间共享和传递数据的一种非常有效的方式.不同进程之间共享的内存通常安排为同一段物理内存. ...

  3. Linux进程间通信——使用共享内存(转)

    一.什么是共享内存 顾名思义,共享内存就是允许两个不相关的进程访问同一个逻辑内存.共享内存是在两个正在运行的进程之间共享和传递数据的一种非常有效的方式.不同进程之间共享的内存通常安排为同一段物理内存. ...

  4. linux进程间通信之共享内存篇

    本文是对http://www.cnblogs.com/andtt/articles/2136279.html中共享内存(上)的进一步阐释说说明 1 共享内存的实现原理 共享内存是linux进程间通讯的 ...

  5. Linux进程间通信(四) - 共享内存

    共享内存的优势 采用共享内存通信的一个显而易见的好处是效率高,因为进程可以直接读写内存,而不需要任何数据的拷贝.对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共享内存则只 ...

  6. linux进程间通信之共享内存学习记录

    进程 狭义定义:进程是正在运行的程序的实例(an instance of a computer program that is being executed). 广义定义:进程是一个具有一定独立功能的 ...

  7. Linux进程间通信之共享内存

    一,共享内存  内核管理一片物理内存,允许不同的进程同时映射,多个进程可以映射同一块内存,被多个进程同时映射的物理内存,即共享内存.  映射物理内存叫挂接,用完以后解除映射叫脱接. 1,共享内存的特点 ...

  8. linux进程间通信同步-共享内存

    参考:https://www.cnblogs.com/charlesblc/p/6142868.html 使用有名信号量,sem_open().sem_close().sem_post().sem_w ...

  9. Linux环境进程间通信(五): 共享内存(上)

    linux下进程间通信的几种主要手段: 管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允 ...

  10. 浅析Linux下进程间通信:共享内存

    浅析Linux下进程间通信:共享内存 共享内存允许两个或多个进程共享一给定的存储区.因为数据不需要在客户进程和服务器进程之间复制,所以它是最快的一种IPC.使用共享内存要注意的是,多个进程之间对一给定 ...

随机推荐

  1. Delphi:ADOConnection连接SQLServer自动断网问题解决

    =============================== 解决方法一:异常时关闭连接,WinXP,win7 32位大部分情况都是起作用的,不过在有些windows操作系统下(如家庭版)不起作用, ...

  2. 深入理解java内置锁(synchronized)和显式锁(ReentrantLock)

    多线程编程中,当代码需要同步时我们会用到锁.Java为我们提供了内置锁(synchronized)和显式锁(ReentrantLock)两种同步方式.显式锁是JDK1.5引入的,这两种锁有什么异同呢? ...

  3. 【python】python sqlalchemy core

    SQLAlchemy是和很多数据库进行交互的一个库,他可以让你创建model,让你可以以一种Python中面向对象的方式进行查询.使得你的代码和数据库可以分开,也就是减轻他们之间的依赖.让你进行数据库 ...

  4. 【bzoj4540】[Hnoi2016]序列 单调栈+离线+扫描线+树状数组区间修改区间查询

    题目描述 给出一个序列,多次询问一个区间的所有子区间最小值之和. 输入 输入文件的第一行包含两个整数n和q,分别代表序列长度和询问数.接下来一行,包含n个整数,以空格隔开,第i个整数为ai,即序列第i ...

  5. BZOJ4710 JSOI2011分特产(容斥原理+组合数学)

    显然可以容斥去掉每人都不为空的限制.每种物品分配方式独立,各自算一个可重组合乘起来即可. #include<iostream> #include<cstdio> #includ ...

  6. hadoop 使用Avro排序

    在上例中,使用Avro框架求出数据的最大值,本例使用Avro对数据排序,输入依然是之前的样本,输出使用文本(也可以输出Avro格式). 1.在Avro的Schema中直接设置排序方向. dataRec ...

  7. windows下vue项目启动步骤

    原创:https://blog.csdn.net/qq_27680317/article/details/71123051?locationNum=10&fps=1 不是ngnix服务器是,忽 ...

  8. MyEclipse下项目的包层次结构调整

    新电脑安装完MyEclipse,导入项目后发现MyEclipse下项目的包层次结构变成了Flat,平面模式,这种模式感觉特别不好, 不能清晰地显示出项目的包层次结构.这样,显示出的包的结构不够明显,我 ...

  9. ubuntu16.04装chrome

    --更简单的方法是先下载chromium浏览器,这是不禁止的,然后打开chromium搜索chrome,chrome的官网下载即可   //安装好后,终端输入google-chrome即可打开 另一种 ...

  10. 8VC Venture Cup 2016 - Final Round (Div. 2 Edition) A

    A. Orchestra time limit per test 2 seconds memory limit per test 256 megabytes input standard input ...