博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Linux环境编程之共享内存区(一):共享内存区简单介绍
阅读量:6340 次
发布时间:2019-06-22

本文共 2262 字,大约阅读时间需要 7 分钟。

共享内存区是可用IPC形式中最快的。一旦内存区映射到共享它的进程的地址空间,进程间数据的传递就不再涉及内核。然而往该共享内存区存放信息或从中取走信息的进程间通常须要某种形式的同步。不再涉及内核是指:进程不再通过运行不论什么进入内核的系统调用来彼此传递数据。内核必须建立同意各个进程共享该内存区的内存映射关系。然后一直管理内存区。

默认情况下通过fork派生的子进程并不与其父进程共享内存区。

mmap函数把一个文件或一个Posix共享内存区对象映射到调用进程的地址空间。使用该函数的目的有:

1、使用普通文件以提供内存映射I/O。

2、使用特殊文件以提供匿名内存映射。

3、使用shm_open以提供无亲缘关系进程间的Posix共享内存区。

#include 
void *mmap(void *addr, size_t len, int prot, int flags, int fd, off_t offset);

addr指定描写叙述符fd应被映射到的进程内空间的起始地址。

若为空指针,则由内核自己去选择起始地址。不管哪种情况。该函数的返回值都是描写叙述符fd所映射到内存区的起始地址。

len是映射到调用进程地址空间中的字节数,它从被映射文件开头起第offset个字节处開始算。

offset通常设置为0。

prot參数指定内存区映射区的保护。

有下面四种:PROT_READ(数据可读)、PROT_WRITE(数据可写)、PROT_EXEC(数据可运行)、PROT_NONE(数据不可訪问)。

flags有下面三种:MAP_SHARED(变动是共享的)、MAP_PRIVATE(变动是私有的)、MAP_FIXED(准确地解释addr參数)。

父子进程之间共享内存区的方法之中的一个是,父进程在调用fork前先指定MAP_SHARED调用mmap。

mmap成功返回后,fd參数能够关闭。该操作对由mmap建立的映射关系没有影响。

为某个进程的地址空间删除一个映射关系,我们调用munmap。

(int munmap(void *addr, size_t len))

调用msync来运行同步。

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

注意:不是全部文件都能进程内存映射,比如:试图把一个訪问终端或套接字的描写叙述符映射到内存将导致mmap返回一个错误。这些类型的描写叙述符必须使用read和write来訪问。

mmap的还有一个用途是在无亲缘关系的进程间提供共享的内存区。

程序实例:

#include 
#include
#include
#include
#include
#include
#include
#define SEM_NAME "mysem"#define FILE_MODE (S_IRUSR|S_IWUSR|S_IRGRP|S_IROTH)int main(int argc, char **argv){ int fd, i, nloop, zero = 0; int *ptr; sem_t *mutex; if(argc != 3){ printf("usage:incrl
<#nloops>\n"); return -1; } nloop = atoi(argv[2]); /*open file , initialize to 0, map into memory*/ fd = open(argv[1], O_RDWR | O_CREAT, FILE_MODE); write(fd, &zero, sizeof(int)); ptr = mmap(NULL, sizeof(int), PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0); close(fd); /*create, initialize, and unlink semaphore*/ mutex = sem_open(SEM_NAME, O_CREAT | O_EXCL, FILE_MODE, 1); if(mutex < 0){ printf("sem_open error.\n"); return -1; } sem_unlink(SEM_NAME); setbuf(stdout, NULL); /*stdout is unbuffered*/ if(fork() == 0){ for(i = 0; i < nloop; i++){ sem_wait(mutex); printf("child:%d\n", (*ptr)++); sem_post(mutex); } exit(0); } /*parent*/ for(i = 0; i < nloop; i++){ sem_wait(mutex); printf("parent:%d\n", (*ptr)++); sem_post(mutex); } exit(0);}
注意:
内存映射一个普通文件时,内存中映射区的大小通常等于该文件的大小。然而文件大小和内存映射区大小能够不同。

转载地址:http://uhhoa.baihongyu.com/

你可能感兴趣的文章
一道面试题引发的关于程序设计的想法
查看>>
2019春第二次课程设计实验报告
查看>>
express入门
查看>>
Codeforces 535D - Tavas and Malekas
查看>>
2017-2018 ACM-ICPC, Asia Tsukuba Regional Contest
查看>>
JS 小总结
查看>>
hihocoder [Offer收割]编程练习赛8
查看>>
[CF98E]Help Shrek and Donkey
查看>>
property函数
查看>>
数论 - 组合数学 + 素数分解 --- hdu 2284 : Solve the puzzle, Save the world!
查看>>
XPath语法
查看>>
linux脚本实现递归阶乘
查看>>
JSON转XML
查看>>
c#典型排序方法总结
查看>>
pkcs1与pkcs8格式RSA私钥互相转换
查看>>
.Net 从零开始构建一个框架之基本实体结构与基本仓储构建
查看>>
C#核编之内建数据类型
查看>>
Oracle运算符收录(易忘记,但是又很重要的运算符)
查看>>
POJ 2062 Card Game Cheater
查看>>
'ascii' codec can't decode byte 0xd6 in position 0
查看>>