文件内存映射,也可以做为进程通信的一种方式,共享一片映射的空间,即可以达到共享数据的要求。
/*
文件映射内存
虚拟内存:把一个文件当做内存使用。如果系统文件中 pagefile.sys,操作系统把这个文件当做内存来操作
这个是技术是把文件映射成内存。
使用景场:如果一个文件很大,而所使用的电脑内存又不够,这时就可以使用此技术,把文件当成内存来
操作。
使用Mapping File提高文件读写的效率。
通过Mapping File在进程间共享内存。
通过文件句柄获得文件路径。
文件映射( mapping)是一种在将文件内容映射到进程的虚拟地址空间的技术。视图(View)是一段虚拟地址
空间,进程可以通过View来存取文件的内容,视图是一段内存,可以使用指针来操作视图。使用的文件映
射之后,读写文件就如同对读写内存一样简单。在使用文件映射时需要创建映射对象,映射对象分为命名
的和未命名的。映射对象还存取权限。
使用文件映射至少有3个好处,一是因为文件是存储于硬盘上的,而文件视图是一段内存,使用文件映射操
作时更方便;二是效率更高;三是可以在不同的进程间共享数据。
*/
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<Windows.h>
char path[1024] = "d:\\demo.txt";
void main()
{
//打开一个已存在的文件,当作映射成内存使用
//如执行成功,则返回文件句柄。INVALID_HANDLE_VALUE表示出错,会设置GetLastError。
/*
函数声明HANDLE CreateFile(LPCTSTR lpFileName, //普通文件名或者设备文件名
DWORD dwDesiredAccess, //访问模式(写/读)
DWORD dwShareMode, //共享模式
LPSECURITY_ATTRIBUTES lpSecurityAttributes, //指向安全属性的指针
DWORD dwCreationDisposition, //如何创建
DWORD dwFlagsAndAttributes, //文件属性
HANDLE hTemplateFile //用于复制文件句柄
*/
//打开一个文件
HANDLE hfile = CreateFileA(path,
GENERIC_READ | GENERIC_WRITE,
0,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL);
if (hfile == INVALID_HANDLE_VALUE)
{
printf("文件打开失败!");
//return;
}
printf("文件大小是:%d\n", GetFileSize(hfile, NULL));
//创建Mapping对象。向操作系统申请,把这个文件当做内存使用
HANDLE hmap = CreateFileMappingA(hfile,
NULL,
PAGE_READWRITE | SEC_COMMIT,//读写,立即提交。输入参数,内存保护属
0,//输入参数,映射大小的最大值的高32位。
GetFileSize(hfile, NULL) + 1,//输入参数,映射大小的最大值的低32位。
NULL);//输入参数,映射对象名,可以是NULL。
//判断对象是否创建成功
if (hmap == NULL)
{
printf("创建失败!\n");
//return;
}
//创建视图,将文件mapping映射到当前进程内存虚拟地址空间
//返回LPVOID值,指向映射的内存值,如果为NULL表示错误
//即把指针指向上面创建内存映射的首地址。
LPVOID pvfile = MapViewOfFile(hmap,
FILE_MAP_READ|FILE_MAP_WRITE,//,存取类别,可以是FILE_MAP_WRITE. FILE_MAP_READ. FILE_MAP_COPY, FILE_MAP_EXECUTE中的一种。
0, //输入参数,映射的文件偏移的高32位。
0, //输入参数,映射的文件偏移的低32位。
0);//输入参数,映射到View的字节数
if (pvfile == NULL)
{
printf("map内存指针映射失败!\n");
CloseHandle(hfile);
CloseHandle(hmap);
//return;
}
puts("映射成功!\n");
char *pstart = pvfile; //首地址
char *pend = pstart + GetFileSize(hfile,NULL); //结束地址
system("pause");
//读出出映射内存的内容
printf("\n开始读取映射内存的内容...\n");
for (int i = 0; i < 1000; i++)
{
putchar(*(pstart + i));
}
//向这片内存写入
char str[156] = "C语言文件映射内存事例!";
//写入时要注意一点是,当文件映射成内存时,是有空间限制的,比如你的文件是1MB,这时
//你要把写的内存指针移到文件尾部这时写入,就会出错。比如这时的指针到了pend,而这时
//写入数据,就会出错。因为这个不属于映射的范围了。
printf("\n开始向映射文件中写入数据...\n");
/*strcpy(pstart, str);*/
//下面这样是写不到数据进内存的。查看文件是没有数据被写入,因为这时他已超过被映射的范围。
char *p = strcpy(pend, str);
if (p == NULL)
{
printf("写入失败!\n");
}
else
{
printf("%s\n", p);
}
system("pause");
UnmapViewOfFile(pvfile);
CloseHandle(hmap);
CloseHandle(hfile);
system("pause");
}