【Windows核心编程】一个使用内存映射文件进行进程间通信的例子

 

进程间通信的方式有很多种,其底层原理使用的都是内存映射文件。

本文实现了Windows核心编程第五版475页上的demo,即使用内存映射文件来在进程间通信。

 

进程1

按钮【Create  mapping of Data】用来创建命名内存映射文件,后备存储器为页交换文件,而非磁盘上的文件,大小为4K,将全部大小映射到进程地址空间,将Data中的数据写入该内存映射文件,然后撤销对文件视图的映射。注意在进程1里不能CloseHandle(m_hFileMapping)

 

进程2

按钮【Open maping and get Data】用来打开进程1创建的命名内存映射文件,将全部大小映射到进程地址空间,从中读出数据,将数据内容放到Data里。 

按钮【Close mapping of Data】用来关闭CloseHandle进程1返回的内存映射文件m_hFileMapping。

 1 void CMemoryMappingDlg::OnBnClickedBtnCreate2()
 2 {
 3     // TODO: 在此添加控件通知处理程序代码;
 4     LPCTSTR lpFileMappingName = _T("MMFSharedData");
 5     m_hFileMapping = CreateFileMapping(
 6         INVALID_HANDLE_VALUE, NULL, PAGE_READWRITE, 0, 4 * 1024, lpFileMappingName);
 7     DWORD dwErr = GetLastError();
 8 
 9     if (NULL == m_hFileMapping )
10     {  
11         AfxMessageBox(_T("无法创建该内存映射文件")); 
12         return; 
13     }
14     if (dwErr == ERROR_ALREADY_EXISTS)
15     {
16         AfxMessageBox(_T("存在同名内存映射文件"));
17         CloseHandle(m_hFileMapping);
18         return;
19     }
20 
21     PVOID pMapOfView = MapViewOfFile(m_hFileMapping, FILE_MAP_READ | FILE_MAP_WRITE, 0, 0, 0 /*4 * 1024*/);
22     if (NULL == pMapOfView)
23     {
24         AfxMessageBox(_T("映射该文件错误")); 
25         CloseHandle(m_hFileMapping);
26         return;
27     }
28 
29     ZeroMemory(pMapOfView, 4 * 1024);
30 
31     CString strText;
32     GetDlgItemText(IDC_EDIT_DATA, strText);
33     memcpy_s(pMapOfView, (strText.GetLength() + 1) * sizeof(TCHAR), strText.GetBuffer(), 
34         (strText.GetLength() + 1) * sizeof(TCHAR));
35 
36     UnmapViewOfFile(pMapOfView);
37     //CloseHandle(m_hFileMapping);  不能执行此句,否则进程2不能打开该命名内存映射文件。
38 }
39 
40 void CMemoryMappingDlg::OnBnClickedBtnClose()
41 {
42     // TODO: 在此添加控件通知处理程序代码;
43     CloseHandle(m_hFileMapping);    
44 }
45 
46 void CMemoryMappingDlg::OnBnClickedBtnOpen()
47 {
48     // TODO: 在此添加控件通知处理程序代码
49     LPCTSTR lpFileMappingName = _T("MMFSharedData"); 
50 
51     //OpenFileMapping的第一个参数一定不能是PAGE_**, 区别于CreateFileMapping函数
52     //HANDLE hFileMapping = OpenFileMapping(PAGE_READWRITE, FALSE, lpFileMappingName);
53     HANDLE hFileMapping = OpenFileMapping(FILE_MAP_READ | FILE_MAP_WRITE, FALSE, lpFileMappingName);
54 
55     if (NULL == hFileMapping)
56     {
57         AfxMessageBox(_T("打不开该内存映射文件"));
58         return;
59     }
60 
61     PVOID pMapOfFile = MapViewOfFile(hFileMapping, FILE_MAP_READ | FILE_MAP_WRITE, 0, 0, 0 /*4 * 1024 */);
62     if (NULL == pMapOfFile)
63     {
64         AfxMessageBox(_T("映射该文件错误"));
65         CloseHandle(hFileMapping);
66         return;
67     }
68 
69     TCHAR tchArr[256];
70     ZeroMemory(tchArr, sizeof(tchArr));
71     memcpy_s(tchArr, sizeof(tchArr), pMapOfFile, sizeof(tchArr));
72     SetDlgItemText(IDC_EDIT_DATA, tchArr);
73 
74     UnmapViewOfFile(pMapOfFile);
75     CloseHandle(hFileMapping);
76 }

 

 

 需要注意的地方是:

0、后备存储器为页交换文件的内存映射文件,CreateFileMapping的文件句柄参数为INVALID_HANDLE_VALUE

1、进程1创建了命名内存映射文件后,一定不能CloseHandle(内存映射文件句柄),否则进程2不能OpenFileMapping打开该命名内存映射文件

2、OpenFileMapping的第一个参数一定不能是PAGE_**, 区别于CreateFileMapping函数

 

 

 

 

 

posted on 2014-04-17 01:21  崔好好  阅读(1750)  评论(0编辑  收藏  举报

导航