一.內(nèi)存映射文件技術(shù)作用
1.使用內(nèi)存映射文件來(lái)訪問(wèn)磁盤上的數(shù)據(jù)文件。這使你可以不必對(duì)文件執(zhí)行 I / O操作,并且可以不必對(duì)文件內(nèi)容進(jìn)行緩存.
2.可以使用內(nèi)存映射文件,使同一臺(tái)計(jì)算機(jī)上運(yùn)行的多個(gè)進(jìn)程能夠相互之間共享數(shù)據(jù)。windows確實(shí)提供了其他一些方法,
以便在進(jìn)程之間進(jìn)行數(shù)據(jù)通信,但是這些方法都是使用內(nèi)存映射文件來(lái)實(shí)現(xiàn)的,諸如使用SendMessage或者PostMessage,
都在內(nèi)部使用了內(nèi)存映射文件.這使得內(nèi)存映射文件成為單個(gè)計(jì)算機(jī)上的多個(gè)進(jìn)程互相進(jìn)行通信的最有效的方法。
二.相當(dāng)函數(shù)介紹
1. CreateFile函數(shù)
(1)作用:創(chuàng)建或打開(kāi)下列對(duì)象,并返回一個(gè)可以用來(lái)訪問(wèn)這些對(duì)象的句柄(可以打開(kāi):文件,管道等)
(2)函數(shù)原型:
HANDLE CreateFile(
LPCTSTR lpFileName, //指向文件名的指針
DWORD dwDesiredAccess, //訪問(wèn)模式(寫/讀)
DWORD dwShareMode, //共享模式
LPSECURITY_ATTRIBUTES lpSecurityAttributes, //指向安全屬性的指針
DWORD dwCreationDisposition, //如何創(chuàng)建
DWORD dwFlagsAndAttributes, //文件屬性
HANDLE hTemplateFile //用于復(fù)制文件句柄
);
(3)參數(shù)介紹

(4)返回值
如執(zhí)行成功,則返回文件句柄。INVALID_HANDLE_VALUE表示出錯(cuò),會(huì)設(shè)置GetLastError。即使函數(shù)成功,
但若文件存在,且指定了CREATE_ALWAYS 或 OPEN_ALWAYS,GetLastError也會(huì)設(shè)為ERROR_ALREADY_EXISTS
2.CreateFileMapping
(1)為指定文件創(chuàng)建一個(gè)有名或無(wú)名的文件映象
(2)函數(shù)原型
HANDLE CreateFileMapping(
HANDLE hFile, // 映射文件的句柄
LPSECURITY_ATTRIBUTES lpFileMappingAttributes,
// 安全描述符指針
DWORD flProtect, // 對(duì)映射對(duì)象的保護(hù)
DWORD dwMaximumSizeHigh, // 對(duì)象最大長(zhǎng)度的高32位
DWORD dwMaximumSizeLow, // 對(duì)象最大長(zhǎng)度的低32位
LPCTSTR lpName // 文件內(nèi)存映射對(duì)象的名字
);
(3)參數(shù)說(shuō)明
hFile:映射文件的句柄,文件的打開(kāi)模式必須與flProtect參數(shù)指定的相一致;如果這個(gè)參數(shù)值為0xFFFFFFFF,
那么必須在dwMaximumSizeHigh和dwMaximumSizeLow參數(shù)中指定映射對(duì)象的大小。并且將在操作系統(tǒng)
虛擬內(nèi)存頁(yè)面替換文件中創(chuàng)建文件映射對(duì)象,而不是使用磁盤文件,同時(shí)必須給出這個(gè)映射對(duì)象的大小。
文件映射對(duì)象通過(guò)副本,遺傳或名字來(lái)共享。
lpFileMappingAttributes:安全描述符指針,決定返回句柄是否能被子進(jìn)程繼承,如果是NULL,那么子進(jìn)程不能
繼承。WinNt中,如果是NULL,那么文件映射對(duì)象得到一個(gè)默認(rèn)的安全描述符。
flProtect:為得到的文件試圖指定保護(hù)模式,可以被設(shè)置為下列值:
PAGE_READONLY :只讀屬性,并且hFile對(duì)應(yīng)的文件必須以GENERIC_READ形式打開(kāi)。
PAGE_READWRITE:可讀可寫屬性,并且hFile對(duì)應(yīng)的文件必須以GENERIC_READ 和 GENERIC_WRITE形式打開(kāi)。
PAGE_WRITECOPY:對(duì)可寫區(qū)域復(fù)制后操作,并且hFile對(duì)應(yīng)的文件必須以GENERIC_READ 和 GENERIC_WRITE形式打開(kāi)。
dwMaximumSizeHigh,dwMaximumSizeLow:如果這兩個(gè)參數(shù)為0,則文件映射對(duì)象的最大長(zhǎng)度等于hFile指定的文件長(zhǎng)度
lpName:文件映射對(duì)象的名字,如果這個(gè)名字已存在,則按照f(shuō)lProtect指定的來(lái)處理映射對(duì)象。如果此參數(shù)為空,則創(chuàng)
建一個(gè)無(wú)名字的文件映射對(duì)象。如果此參數(shù)的名字與系統(tǒng)事件的名字相同,則函數(shù)執(zhí)行失敗,GetLastError返回
ERROR_INVALID_HANDLE;
(4)返回值
數(shù)調(diào)用成功返回文件映射對(duì)象的句柄,如果文件映射對(duì)象已經(jīng)存在則返回原有映射對(duì)象的句柄,GetLastError返回
ERROR_ALREADY_EXISTS。函數(shù)執(zhí)行失敗返回Null
3. MapViewOfFile
(1)在調(diào)用進(jìn)程的地址空間映射一個(gè)文件視圖
(2)函數(shù)原型
LPVOID MapViewOfFile(
HANDLE hFileMappingObject, // 已創(chuàng)建的文件映射對(duì)象句柄
DWORD dwDesiredAccess, // 訪問(wèn)模式
DWORD dwFileOffsetHigh, // 文件偏移的高32位
DWORD dwFileOffsetLow, // 文件偏移的低32位
DWORD dwNumberOfBytesToMap // 映射視圖的大小
);
(3)參數(shù)
hFileMappingObject: 由CreateFileMapping 或 OpenFileMapping 返回的文件映射對(duì)象句柄。
dwDesiredAccess: 映射視圖的訪問(wèn)模式,與創(chuàng)建文件映射對(duì)象的保護(hù)模式flProtect有關(guān),可以被設(shè)置為下列值:
FILE_MAP_WRITE: 一個(gè)可讀寫屬性的文件視圖被創(chuàng)建,保護(hù)模式為PAGE_READWRITE
FILE_MAP_READ : 一個(gè)只讀屬性的文件視圖被創(chuàng)建,保護(hù)模式為PAGE_READWRITE 或 PAGE_READONLY
FILE_MAP_ALL_ACCESS:與FILE_MAP_WRITE模式相同
FILE_MAP_COPY: 保護(hù)模式為PAGE_WRITECOPY時(shí),得到一個(gè)視圖文件,當(dāng)你對(duì)視圖文件寫操作時(shí),頁(yè)面自動(dòng)
交換,并且你所做的修改不會(huì)損壞原始數(shù)據(jù)資料。
dwNumberOfBytesToMap:映射文件部分的大小,如果為0,則映射整個(gè)文件。
(4)返回值:如果成功返回返回映射視圖的起始地址,如果失敗返回NULL
4.UnmapViewOfFile
(1)刪除文件的映射視圖
(2)函數(shù)原型
BOOL UnmapViewOfFile(
LPCVOID lpBaseAddress // 映射視圖起始地址
);
(3)參數(shù)
lpBaseAddress:映射視圖起始地址,由 MapViewOfFile 函數(shù) MapViewOfFileEx產(chǎn)生
(4)返回值:如果調(diào)用成功返回非零,并且所有指定地址內(nèi)的臟頁(yè)面會(huì)被寫入硬盤。調(diào)用失敗返回零。
5.FlushViewOfFile
(1)把文件映射視圖中的修改的內(nèi)容或全部寫回到磁盤文件中
(2)函數(shù)原型
BOOL FlushViewOfFile(
LPCVOID lpBaseAddress, // 修改內(nèi)容的起始地址
DWORD dwNumberOfBytesToFlush // 修改的字節(jié)數(shù)目
);
(3)返回值:函數(shù)執(zhí)行成功返回非零。
6.MapViewOfFileEx
(1)在調(diào)用進(jìn)程的地址空間映射一個(gè)文件視圖,并且允許調(diào)用進(jìn)程為映射視圖指定特殊的內(nèi)存地址
(2)函數(shù)原型:
LPVOID MapViewOfFileEx(
HANDLE hFileMappingObject, // 文件映射對(duì)象的句柄
DWORD dwDesiredAccess, // 訪問(wèn)模式
DWORD dwFileOffsetHigh, // 文件偏移的高32位
DWORD dwFileOffsetLow, // 文件偏移的低32位
DWORD dwNumberOfBytesToMap, // 映射視圖的大小
LPVOID lpBaseAddress // 指定映射視圖的其實(shí)內(nèi)存地址
);
注意:
與MapViewOfFile用法相同,但是如果指定的內(nèi)存地址空間大小不夠,則函數(shù)執(zhí)行失敗。
7.OpenFileMapping
(1)打開(kāi)一個(gè)已命名的文件映射對(duì)象
(2)函數(shù)原型
HANDLE OpenFileMapping(
DWORD dwDesiredAccess, // 訪問(wèn)模式
BOOL bInheritHandle, // 繼承標(biāo)志
LPCTSTR lpName // 文件映射對(duì)象名指針
);
(3)參數(shù)
dwDesiredAccess:訪問(wèn)模式與MapViewOfFile中的訪問(wèn)模式相同。
bInheritHandle:繼承標(biāo)志,是否可以被一個(gè)新的進(jìn)程繼承使用,如果為TRUE,就可以被一個(gè)新進(jìn)程繼承句柄。
(4)返回值
成功返回一個(gè)已命名的文件映射對(duì)象,失敗返回NULL。
三. 以下是簡(jiǎn)單的操作示例
#include <Windows.h>
#include <conio.h>
#include <stdlib.h>
#include <stdio.h>

int main(int argc,char argv[])


{
HANDLE hFile;
OFSTRUCT opBuf;
HANDLE hMapfile;
HANDLE hMapview;
char *recv;

//1.創(chuàng)建或打開(kāi)文件
hFile = ::CreateFile("D:\\MapFileTest.txt", GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ|FILE_SHARE_WRITE, NULL,OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
if (hFile==0)

{
printf("open file failed!\n");
return 1;
}
//2.為指定文件創(chuàng)建一個(gè)有名的文件映象
hMapfile=CreateFileMapping((HANDLE)hFile,NULL,PAGE_READWRITE,0,3,"MapTest");
if(hMapfile==NULL)

{
printf("mapping file failed!\n");
return 1;
}
//關(guān)閉文件句柄
CloseHandle((HANDLE)hFile);
hFile=0;
//3.在調(diào)用進(jìn)程的地址空間映射一個(gè)文件視圖
hMapview=MapViewOfFile(hMapfile,FILE_MAP_WRITE,0,0,0);
if(hMapview==NULL)

{
printf("mapping view failed!\n");
return 1;
}

//往文件映射視圖中寫入數(shù)據(jù)
recv=(char *)hMapview;
printf("Mapping view's content is :%.10s \n",recv);
strcpy(recv,"abc");

//把文件映射視圖中的修改的內(nèi)容或全部寫回到磁盤文件中
FlushViewOfFile(recv,3);

//4.刪除文件的映射視圖
printf("Mapping view's content is :%.10s \n",recv);
UnmapViewOfFile(hMapview);

//關(guān)閉文件映象句柄
CloseHandle(hMapfile);
hMapfile = 0;


_getch();
return 0;
}
