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. bzoj1835[ZJOI2010]基站选址

    主席树+决策单调,重写一遍比之前短多了……题解:http://www.cnblogs.com/liu-runda/p/6051422.html #include<cstdio> #incl ...

  2. Redis的RDB和AOF持久化

    RDB 持久化:在指定的时间间隔内生成数据集的时间点快照. AOF 持久化:记录服务器执行的所有写操作命令,并在服务器启动时,通过重新执行这些命令来还原数据集. RDB 它只保存了 Redis 在某个 ...

  3. BZOJ2395:[Balkan 2011]Timeismoney——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=2395 有n个城市(编号从0..n-1),m条公路(双向的),从中选择n-1条边,使得任意的两个城市 ...

  4. BZOJ1833:[ZJOI2010]数字计数——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=1833 https://www.luogu.org/problemnew/show/P2602 给定两 ...

  5. BZOJ1013:[JSOI2008]球形空间产生器——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=1013 Description 有一个球形空间产生器能够在n维空间中产生一个坚硬的球体.现在,你被困在 ...

  6. Linux实验一

    一.Linux 简介 1.Linux 就是一个操作系统,就像你多少已经了解的 Windows(xp,7,8)和 Max OS , 我们的 Linux 也就是系统调用和内核那两层,当然直观的来看,我们使 ...

  7. UVA.11300 Spreading the Wealth (思维题 中位数模型)

    UVA.11300 Spreading the Wealth (思维题) 题意分析 现给出n个人,每个人手中有a[i]个数的金币,每个人能给其左右相邻的人金币,现在要求你安排传递金币的方案,使得每个人 ...

  8. ContestHunter暑假欢乐赛 SRM 03

    你们也没人提醒我有atcoderQAQ... A题曼哈顿距离=欧拉距离就是在同一行或者同一列,记录下i,j出现过的次数,减去就行,直接map过. B题一开始拿衣服了,一直以为排序和不排序答案是一个样的 ...

  9. HDU 5249 离线树状数组求第k大+离散化

    KPI Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...

  10. HDU2121:Ice_cream’s world II (虚根+有向图最小生成树)

    Ice_cream’s world II Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Oth ...