CreateFileMapping和MapViewOfFile

在开发软件过程里,也经常碰到进程间共享数据的需求。比如A进程创建计算数据,B进程进行显示数据的图形。这样的开发方式可以把一个大程序分开成独立的小程序,提高软件的成功率,也可以更加适合团队一起开发,加快软件的开发速度。下面就来使用文件映射的方式进行共享数据。先要使用函数CreateFileMapping来创建一个想共享的文件数据句柄,然后使用MapViewOfFile来获取共享的内存地址,然后使用OpenFileMapping函数在另一个进程里打开共享文件的名称,这样就可以实现不同的进程共享数据。 函数CreateFileMapping、MapViewOfFile声明如下:WINBASEAPI__outHANDLEWINAPICreateFileMappingA(    __in     HANDLE hFile,    __in_opt LPSECURITY_ATTRIBUTES lpFileMappingAttributes,    __in     DWORD flProtect,    __in     DWORD dwMaximumSizeHigh,    __in     DWORD dwMaximumSizeLow,    __in_opt LPCSTR lpName    );

WINBASEAPI__outHANDLEWINAPICreateFileMappingW(    __in     HANDLE hFile,    __in_opt LPSECURITY_ATTRIBUTES lpFileMappingAttributes,    __in     DWORD flProtect,    __in     DWORD dwMaximumSizeHigh,    __in     DWORD dwMaximumSizeLow,    __in_opt LPCWSTR lpName    );

#ifdef UNICODE

#define CreateFileMapping CreateFileMappingW

#else

#define CreateFileMapping CreateFileMappingA

#endif // !UNICODE WINBASEAPI__outLPVOIDWINAPIMapViewOfFile(    __in HANDLE hFileMappingObject,    __in DWORD dwDesiredAccess,    __in DWORD dwFileOffsetHigh,    __in DWORD dwFileOffsetLow,    __in SIZE_T dwNumberOfBytesToMap    );

hFile是创建共享文件的句柄。lpFileMappingAttributes是文件共享的属性。flProtect是当文件映射时读写文件的属性。dwMaximumSizeHigh是文件共享的大小高位字节。dwMaximumSizeLow是文件共享的大小低位字节。lpName是共享文件对象名称。hFileMappingObject是共享文件对象。dwDesiredAccess是文件共享属性。dwFileOffsetHigh是文件共享区的偏移地址。dwFileOffsetLow是文件共享区的偏移地址。dwNumberOfBytesToMap是共享数据长度。 调用函数的例子如下:

#001  //文件共享。

#002  //蔡军生 2007/10/27 QQ:9073204 深圳

#003  void FileMapping(void)

#004  {

#005         //打开共享的文件对象。

#006         m_hMapFile = OpenFileMapping(FILE_MAP_ALL_ACCESS,

#007              FALSE,_T("TestFileMap"));

#008         if (m_hMapFile)

#009         {

#010               //显示共享的文件数据。

#011              LPTSTR lpMapAddr = (LPTSTR)MapViewOfFile(m_hMapFile,FILE_MAP_ALL_ACCESS,

#012                   0,0,0);

#013               OutputDebugString(lpMapAddr);

#014         }

#015         else

#016         {

#017               //创建共享文件。

#018               m_hMapFile = CreateFileMapping( (HANDLE)0xFFFFFFFF,NULL,

#019                   PAGE_READWRITE,0,1024,_T("TestFileMap"));

#020 

#021               //拷贝数据到共享文件里。

#022               LPTSTR lpMapAddr = (LPTSTR)MapViewOfFile(m_hMapFile,FILE_MAP_ALL_ACCESS,

#023                    0,0,0);

#024               std::wstring strTest(_T("TestFileMap"));

#025               wcscpy(lpMapAddr,strTest.c_str());                

#026 

#027               FlushViewOfFile(lpMapAddr,strTest.length()+1);

#028         }

#029  } 引用: http://blog.csdn.net/caimouse/archive/2007/10/27/1848910.aspx 

内存映射API函数CreateFileMapping创建一个有名的共享内存:
HANDLE CreateFileMapping(
HANDLE hFile,                                                                        // 映射文件的句柄,
                                                                                                   //设为0xFFFFFFFF以创建一个进程间共享的对象
LPSECURITY_ATTRIBUTES lpFileMappingAttributes,   // 安全属性
DWORD flProtect,                                                                   // 保护方式
DWORD dwMaximumSizeHigh,                                           //对象的大小 
DWORD dwMaximumSizeLow, 
LPCTSTR lpName                                                                 // 必须为映射文件命名
);

与虚拟内存类似,保护方式可以是PAGE_READONLY或是PAGE_READWRITE。如果多进程都对同一共享内存进行写访问,则必须保持相互间同步。映射文件还可以指定PAGE_WRITECOPY标志,可以保证其原始数据不会遭到破坏,同时允许其他进程在必要时自由的操作数据的拷贝。

在创建文件映射对象后使用可以调用MapViewOfFile函数映射到本进程的地址空间内。

下面说明创建一个名为MySharedMem的长度为4096字节的有名映射文件:
HANDLE hMySharedMapFile=CreateFileMapping((HANDLE)0xFFFFFFFF),
NULL,PAGE_READWRITE,0,0x1000,"MySharedMem");
并映射缓存区视图:
LPSTR pszMySharedMapView=(LPSTR)MapViewOfFile(hMySharedMapFile,
FILE_MAP_READ|FILE_MAP_WRITE,0,0,0);

其他进程访问共享对象,需要获得对象名并调用OpenFileMapping函数。
HANDLE hMySharedMapFile=OpenFileMapping(FILE_MAP_WRITE,
FALSE,"MySharedMem");

一旦其他进程获得映射对象的句柄,可以象创建进程那样调用MapViewOfFile函数来映射对象视图。用户可以使用该对象视图来进行数据读写操作,以达到数据通讯的目的。

当用户进程结束使用共享内存后,调用UnmapViewOfFile函数以取消其地址空间内的视图:
if (!UnmapViewOfFile(pszMySharedMapView))
{

         AfxMessageBox("could not unmap view of file");

 }

 

void CreateFileMappingEx() 

DWORD timebegin = ::timeGetTime(); 
HANDLE fp = CreateFile(TEXT("E://jyzhj2.rar"),//这里输入需要复制的文件 src 
GENERIC_READ | GENERIC_WRITE, 
FILE_SHARE_READ, 
NULL, 
OPEN_EXISTING, 
FILE_FLAG_SEQUENTIAL_SCAN, 
NULL); 

if(fp == NULL) 

    cout<<"错误"<<endl; 
    return; 


DWORD dwBytesInBlock = GetFileSize(fp,NULL); //文件长度 

// 创建文件映射内核对象,句柄保存于hFileMapping 
HANDLE hFileMapping = CreateFileMapping(fp, 
    NULL, 
    PAGE_READWRITE, 
    0,//(DWORD)(dwBytesInBlock >> 16), 
    dwBytesInBlock,//(DWORD)(dwBytesInBlock & 0x0000FFFF), 
    NULL); 

int dwError = GetLastError(); 

// 释放文件内核对象 
CloseHandle(fp); 

// 偏移地址 
__int64 qwFileOffset = 0; 

// 将文件数据映射到进程的地址空间 
LPVOID pbFile = (LPVOID)MapViewOfFile( hFileMapping, 
    FILE_MAP_ALL_ACCESS, 
    (DWORD)(qwFileOffset >> 32), 
    (DWORD)(qwFileOffset & 0xFFFFFFFF), 
    dwBytesInBlock); 

HANDLE wp = CreateFile( TEXT("E://仙剑5.rar"),//这里输入 需要粘贴的文件 dst 
    GENERIC_READ | GENERIC_WRITE, 
    FILE_SHARE_WRITE, 
    NULL, 
    CREATE_ALWAYS, 
    FILE_FLAG_SEQUENTIAL_SCAN | FILE_FLAG_WRITE_THROUGH, 
    NULL); 

HANDLE hFileMapping2 = CreateFileMapping( wp, 
    NULL, 
    PAGE_READWRITE, 
    0,//(DWORD)(dwBytesInBlock >> 16), 
    dwBytesInBlock,//(DWORD)(dwBytesInBlock & 0x0000FFFF), 
    NULL); 


CloseHandle(wp); 

LPVOID pbFile2 = (LPVOID)MapViewOfFile( hFileMapping2, 
    FILE_MAP_ALL_ACCESS, 
    (DWORD)(qwFileOffset >> 32), 
    (DWORD)(qwFileOffset & 0xFFFFFFFF), 
    dwBytesInBlock); 

memcpy(pbFile2,pbFile,dwBytesInBlock); 

UnmapViewOfFile(pbFile2); 
UnmapViewOfFile(pbFile); 

CloseHandle(hFileMapping2); 
CloseHandle(hFileMapping); 
DWORD timeend = ::timeGetTime(); 
cout<<"CreateFileMapping和MapViewOfFile程序运行时间为"<<timeend - timebegin<<endl; 


原文网址:http://hi.baidu.com/y_x_b_s/item/dc8e2beca86ee7c5bbf37dc8

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值