1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
|
#include<stdio.h>
#include<stdlib.h>
#include<sys/mman.h>
#include<unistd.h>
#include<fcntl.h>
#define NumReconds 100
typedef
struct
{
int
iNum;
char
sName[24];
} Recond;
int
main(
void
)
{
Recond recond,*mapped;
int
i,f;
FILE
*fp;
fp=
fopen
(
"recond.dat"
,
"w+"
);
for
( i=0; i < NumReconds; i++)
{
recond.iNum = i;
sprintf
(recond.sName,
"Recond-%d\n"
,i);
fwrite
(&recond,
sizeof
(Recond),1,fp);
}
fclose
(fp);
//使用传统方式修改文件内容
fp =
fopen
(
"recond.dat"
,
"r+"
);
//获得要修改文件的位置
fseek
(fp,43*
sizeof
(recond),SEEK_SET);
fread
(&recond,
sizeof
(recond),1,fp);
recond.iNum = 143;
sprintf
(recond.sName,
"Recond-%d"
,recond.iNum);
fwrite
(&recond,
sizeof
(recond),1,fp);
fclose
(fp);
//使用内存映射的方式打开文件,修改文件内存
//注意这里是open打开不是fopen!!!!
f = open(
"recond.dat"
,O_RDWR);
//获得磁盘文件的内存映射
mapped = (Recond *) mmap(0 , NumReconds *
sizeof
(Recond) , PROT_READ|PROT_WRITE, MAP_SHARED, f, 0);
mapped[43].iNum = 999;
sprintf
(mapped[43].sName,
"Recond-%d"
,mapped[43].iNum);
//将修改同步到磁盘中
msync((
void
*)mapped,NumReconds*
sizeof
(recond),MS_ASYNC);
//关闭内存映射
munmap((
void
*)mapped,NumReconds*
sizeof
(recond));
close(f);
exit
(0);
}
|
void *mmap(void *start, size_t length, int prot, int flags,
start:映射区的开始地址,设置为0时表示由系统决定映射区的起始地址。
length:映射区的长度。//长度单位是 以字节为单位,不足一内存页按一内存页处理
prot:期望的内存保护标志,不能与文件的打开模式冲突。是以下的某个值,可以通过or运算合理地组合在一起
PROT_EXEC //页内容可以被执行
PROT_READ //页内容可以被读取
PROT_WRITE //页可以被写入
PROT_NONE //页不可访问
MAP_FIXED //使用指定的映射起始地址,如果由start和len参数指定的内存区重叠于现存 映射空间,重叠部分将会被丢弃。如果指定的起始地址不可用,操作将会失败。并且起始地 址必须落在页的边界上。
MAP_SHARED //与其它所有映射这个对象的进程共享映射空间。对共享区的写入,相当于输出到文件。直到msync()或者munmap()被调用,文件实际上不会被更新。
MAP_PRIVATE //建立一个写入时拷贝的私有映射。内存区域的写入不会影响到原文件。这个标志和以上标志是互斥的,只能使用其中一个。
MAP_DENYWRITE //这个标志被忽略。
fd:有效的文件描述词。一般是由open()函数返回,其值也可以设置为-1,此时需要指定flags参数中的MAP_ANON,表明进行的是匿名映射。
进程在映射空间的对共享内容的改变并不直接写回到磁盘文件中,往往在调用munmap()后才执行该操作。
flags:刷新的参数设置,可以取值MS_ASYNC/ MS_SYNC/ MS_INVALIDATE
其中:
取值为MS_ASYNC(异步)时,调用会立即返回,不等到更新的完成;
取值为MS_SYNC(同步)时,调用会等到更新完成之后返回;