文件内存映射,也可以做为进程通信的一种方式,共享一片映射的空间,即可以达到共享数据的要求。

/* 
文件映射内存
虚拟内存:把一个文件当做内存使用。如果系统文件中 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");
}