概述

共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据(如图)。

共享内存 VS. 其他IPC形式

用管道/消息队列传递数据

用共享内存传递数据

共享内存生成之后,传递数据并不需要再走Linux内核,共享内存允许两个或多个进程共享一个给定的存储区域,数据并不需要在多个进程之间进行复制,因此,共享内存的传输速度更快!

mmap内存映射

将文件/设备空间映射到共享内存区

#include <sys/mman.h>
void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offset);
int munmap(void *addr, size_t length);

参数:

addr:  要映射的起始地址, 通常指定为NULL, 让内核自动选择;

length: 映射到进程地址空间的字节数;

prot: 映射区保护方式(见下);

flags: 标志(见下);

fd: 文件描述符;

offset: 从文件头开始的偏移量;

prot

说明

PROT_READ

页面可读

PROT_WRITE

页面可写

PROC_EXEC

页面可执行

PROC_NONE

页面不可访问

flags

说明

MAP_SHARED

变动是共享的

MAP_PRIVATE

变动是私有的

MAP_FIXED

准确解释addr参数, 如果不指定该参数, 则会以4K大小的内存进行对齐

MAP_ANONYMOUS

建立匿名映射区, 不涉及文件

mmap返回值:

成功: 返回映射到的内存区的起始地址;

失败: 返回MAP_FAILED;

内存映射示意图:

(注意: 内存映射时, 是以页面(4K)作为单位)

/** 示例1: 写文件映射
将文件以可读,可写的方式映射到进程的地址空间, 然后向其中写入内容
**/
struct Student
{char name[4];int age;
};
int main(int argc,char **argv)
{if (argc != 2)err_quit("usage: ./main <file-name>");int fd = open(argv[1], O_CREAT|O_RDWR|O_TRUNC, 0666);if (fd == -1)err_exit("file open error");//为内存映射争取空间if (lseek(fd, sizeof(Student)*5-1, SEEK_SET) == (off_t)-1)err_exit("lseek error");write(fd, "", 1);Student *p = (Student *)mmap(NULL, sizeof(Student)*5,PROT_WRITE|PROT_READ, MAP_SHARED, fd, 0);if (p == MAP_FAILED)err_exit("mmap error");// 此时:操纵文件就可以如同操纵内存一样了char ch = 'a';for (int i = 0; i < 5; ++i){memcpy((p+i)->name, &ch, 1);(p+i)->age = 20+i;++ ch;}cout << "file initialized!" << endl;if (munmap(p, sizeof(Student)*5) == -1)err_exit("munmap error");cout << "process exit..." << endl;return 0;
}
/**示例2: 读文件映射
**/
int main(int argc,char **argv)
{if (argc != 2)err_quit("usage: ./main <file-name>");//以只读方式打开int fd = open(argv[1], O_RDONLY);if (fd == -1)err_exit("file open error");void *mmap(void *addr, size_t length, int prot, int flags,int fd, off_t offset);Student *p = (Student *)mmap(NULL, sizeof(Student)*5,PROT_READ, MAP_SHARED, fd, 0);if (p == MAP_FAILED)err_exit("mmap error");// 从内存中读取数据(其实是从文件中读取)for (int i = 0; i < 5; ++i){cout << "name: " << (p+i)->name << ", age: " << (p+i)->age << endl;}if (munmap(p, sizeof(Student)*5) == -1)err_exit("munmap error");cout << "process exit..." << endl;return 0;
}

map注意点:

1. 内存映射不能(也不可能)改变文件的大小;

2. 可用于进程间通信的有效地址空间不完全受限于映射文件的大小, 而应该以内存页面的大小为准(见下面测试);

3. 文件一旦被映射之后, 所有对映射区域的访问实际上是对内存区域的访问; 映射区域内容写会文件时, 所写内容不能超过文件的大小.

/** 测试: 注意点2
文件以40K的内容进行创建, 而以120K的内容进行写回
**/
//程序1: 写文件映射
int main(int argc,char **argv)
{if (argc != 2)err_quit("usage: ./main <file-name>");int fd = open(argv[1], O_CREAT|O_RDWR|O_TRUNC, 0666);if (fd == -1)err_exit("file open error");// 注意: 此处我们的文件其实只有40个字节if (lseek(fd, sizeof(Student)*5-1, SEEK_SET) == (off_t)-1)err_exit("lseek error");write(fd, "", 1);// 但是我们却是以120个字节的方式进行映射Student *p = (Student *)mmap(NULL, sizeof(Student)*15,PROT_WRITE|PROT_READ, MAP_SHARED, fd, 0);if (p == MAP_FAILED)err_exit("mmap error");// 以120个字节的方式进行写入char ch = 'a';for (int i = 0; i < 15; ++i){memcpy((p+i)->name, &ch, 1);(p+i)->age = 20+i;++ ch;}cout << "file initialized!" << endl;// 以120字节的方式卸载该内存区if (munmap(p, sizeof(Student)*15) == -1)err_exit("munmap error");// 注意: 要故意暂停一会儿, 以便让read程序读取该共享内存的内容sleep(20);cout << "process exit..." << endl;
}
//程序2: 读文件映射
int main(int argc,char **argv)
{if (argc != 2)err_quit("usage: ./main <file-name>");//以只读方式打开int fd = open(argv[1], O_RDONLY);if (fd == -1)err_exit("file open error");void *mmap(void *addr, size_t length, int prot, int flags,int fd, off_t offset);// 以120字节的方式映射Student *p = (Student *)mmap(NULL, sizeof(Student)*15,PROT_READ, MAP_SHARED, fd, 0);if (p == MAP_FAILED)err_exit("mmap error");// 以120字节的方式读取for (int i = 0; i < 15; ++i){cout << "name: " << (p+i)->name << ", age: " << (p+i)->age << endl;}if (munmap(p, sizeof(Student)*15) == -1)err_exit("munmap error");cout << "process exit..." << endl;
}

msync函数

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

对映射的共享内存执行同步操作

参数:

addr: 内存起始地址;

length: 长度

flags: 选项

flags

说明

MS_ASYNC

执行异步写

MS_SYNC

执行同步写, 直到内核将数据真正写入磁盘之后才返回

MS_INVALIDATE

使高速缓存的数据失效

返回值:

成功: 返回0;

失败: 返回-1;

Linux IPC实践(8) --共享内存/内存映射相关推荐

  1. Linux IPC实践(10) --Posix共享内存

    1. 创建/获取一个共享内存 #include <sys/mman.h> #include <sys/stat.h> /* For mode constants */ #inc ...

  2. Linux IPC实践(9) --System V共享内存

    共享内存API #include <sys/ipc.h> #include <sys/shm.h>int shmget(key_t key, size_t size, int ...

  3. Linux IPC实践(13) --System V IPC综合实践

    实践:实现一个先进先出的共享内存shmfifo 使用消息队列即可实现消息的先进先出(FIFO), 但是使用共享内存实现消息的先进先出则更加快速; 我们首先完成C语言版本的shmfifo(基于过程调用) ...

  4. Linux IPC实践(11) --System V信号量(1)

    信号量API #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h> int semget ...

  5. Linux IPC实践(7) --Posix消息队列

    1. 创建/获取一个消息队列 #include <fcntl.h> /* For O_* constants */ #include <sys/stat.h> /* For m ...

  6. Linux IPC实践(4) --System V消息队列(1)

    消息队列概述 消息队列提供了一个从一个进程向另外一个进程发送一块数据的方法(仅局限于本机); 每个数据块都被认为是有一个类型,接收者进程接收的数据块可以有不同的类型值. 消息队列也有管道一样的不足:  ...

  7. Linux IPC实践(2) --匿名PIPE

    管道概念 管道是Unix中最古老的进程间通信的形式,我们把从一个进程连接到另一个进程的一个数据流称为一个"管道", 管道的本质是固定大小的内核缓冲区; 如:ps aux | gre ...

  8. Linux IPC实践(6) --System V消息队列(3)

    消息队列综合案例 消息队列实现回射客户/服务器 server进程接收时, 指定msgtyp为0, 从队首不断接收消息 server进程发送时, 将mtype指定为接收到的client进程的pid cl ...

  9. Linux IPC实践(12) --System V信号量(2)

    实践1:信号量实现进程互斥 父子进程执行流程如下: 父进程 子进程 P P O(print) X(print) sleep sleep O(print) X(print) V V sleep slee ...

最新文章

  1. 三个大数据处理框架:Storm,Spark和Samza 介绍比较
  2. gulp复制整个文件夹或文件到指定目录(包括拷贝单个文件)
  3. java元婴期(23)----java进阶(mybatis(2)---mapper代理mybatis核心配置文件输入输出映射)
  4. css-modules 简介
  5. kafka数据到flume_大数据摄取:Flume,Kafka和NiFi
  6. 从零开始学视觉Transformer(3):视觉问题中的注意力机制
  7. LeetCode 94. 二叉树的中序遍历(中序遍历)
  8. angular中如何定义全局变量_如何在Angular 2 / Typescript中声明全局变量?
  9. SIMIS计算机联锁系统,steam游戏sim
  10. 从今天开始 好好规划自己
  11. php获取笔顺矢量,php如何获取汉字笔画数功能的实例分析
  12. VITS 语音合成完全端到端TTS的里程碑
  13. Unity webGl 鼠标手指触屏控制相机围绕物体 360度旋转
  14. 心知天气天气状况获取,ESP32获取天气信息(含源码)
  15. mysql implode_php implode函数应用
  16. 卡特兰数(c++实现)
  17. 什么是NP问题,什么是NP hard问题,什么是NP完全问题。
  18. 【ArangoDB 介绍】
  19. 论文阅读:CNN+GCN
  20. 达内code下载地址

热门文章

  1. 在jsp页面中实现格式化数字,百分比,货币
  2. [LintCode] 字符串查找
  3. EasyUI框架入门学习
  4. Ubuntu下 Oracle sqldeveloper中文目录、文件,select查询结果中:中文乱码
  5. 原始代理需要改进的地方
  6. NVelocity模板引擎初学总结。[zhuan]
  7. Distance计算的距离随经纬度不同
  8. mysql的数据现实在小程序_使用phpstudy将本地mysql数据显示在微信小程序前端
  9. window连接不上linux ftp_Linux文件自动备份方案
  10. sqlite库——c语言实现匹配已知字符串中某个字段(该字段在其他表中),在其他表中获取值并显示