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. [剑指Offer] 52.正则表达式匹配

    题目描述 请实现一个函数用来匹配包括'.'和'*'的正则表达式.模式中的字符'.'表示任意一个字符,而'*'表示它前面的字符可以出现任意次(包含0次). 在本题中,匹配是指字符串的所有字符匹配整个模式 ...

  2. SpringBoot Web(SpringMVC)

    入门工程: package com.example.demo.controller; import com.example.demo.entity.User; import org.springfra ...

  3. Matlab中fspecial的用法【转】

    Fspecial函数用于创建预定义的滤波算子,其语法格式为:h = fspecial(type)h = fspecial(type,parameters,sigma) 参数type制定算子类型,par ...

  4. RT-thread内核之空闲线程

    空闲线程是系统线程中一个比较特殊的线程,它具有最低的优先级,当系统中无其他线程可运行时,调度器将调度到空闲线程.空闲线程通常是一个死循环,永远不被挂起.RT-Thread实时操作系统为空闲线程提供了钩 ...

  5. 【codevs1282】约瑟夫问题 Treap

    题目描述 有编号从1到N的N个小朋友在玩一种出圈的游戏.开始时N个小朋友围成一圈,编号为I+1的小朋友站在编号为I小朋友左边.编号为1的小朋友站在编号为N的小朋友左边.首先编号为1的小朋友开始报数,接 ...

  6. Gevent-socket

    1. 通过Gevent实现单线程下的多socket并发. server 端: #server side import sys import socket import time import geve ...

  7. BZOJ1031:[JSOI2007]字符加密——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=1031 喜欢钻研问题的JS同学,最近又迷上了对加密方法的思考.一天,他突然想出了一种他认为是终极的加密 ...

  8. BZOJ4815 [CQOI2017]小Q的表格 【数论 + 分块】

    题目链接 BZOJ4815 题解 根据题中的式子,手玩一下发现和\(gcd\)很像 化一下式子: \[ \begin{aligned} bf(a,a + b) &= (a + b)f(a,b) ...

  9. Redux的应该注意的问题

    1. Store中的State修改不能直接修改原有的State,若直接修改State,则redux中的所有操作都将指向 内存中的同一个state,将无法获取每一次操作前后的state,就无法追溯sta ...

  10. 基于DCT系数的实时监控中运动目标检测

    本文的主要内容来自2009 Advanced Video and Signal Based Surveillance会议的一篇论文“Real-Time Moving Object Detection ...