win32下进程间通信(共享内存)实例分析

一、概述

很多情况下在Windows程序中,各个进程之间往往需要交换数据,进行数据通讯。WIN32 API提供了许多函数使我们能够方便高效的进行进程间的通讯,通过这些函数我们可以控制不同进程间的数据交换。

进程间通讯(即:同机通讯)和数据交换有多种方式:消息、共享内存、匿名(命名)管道、邮槽、Windows套接字等多种技术。“共享内存”(shared memory)可以定义为对一个以上的进程是可见的内存或存在于多个进程的虚拟地址空间。例如:如果两个进程使用相同的DLL,只把DLL的代码页装入内存一次,其他所有映射这个DLL的进程只要共享这些代码页就可以了;利用消息机制实现IPC虽然有交换的数据量小、携带的信息少等缺点,但由于其实现方便、应用灵活而广泛应用于无须大量、频繁数据交换的内部进程通讯系统之中。

二、同机进程间共享内存的实现

采用内存映射文件实现WIN32进程间的通讯:Windows中的内存映射文件的机制为我们高效地操作文件提供了一种途径,它允许我们在WIN32进程中保留一段内存区域,把硬盘或页文件上的目标文件映射到这段虚拟内存中。注意:在程序实现中必须考虑各进程之间的同步问题。

具体实现步骤如下:

1、在服务器端进程中调用内存映射API函数CreateFileMapping创建一个有名字标识的共享内存;

函数CreateFileMapping原型如下:

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

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

例如:创建一个名为“zzj”的长度为4096字节的有名映射文件:

HANDLE m_hMapFile=CreateFileMapping((HANDLE)0xFFFFFFFF,NULL,PAGE_READWRITE,0,0x1000," zzj");

2、在创建文件映射对象后,服务器端进程调用MapViewOfFile函数映射到本进程的地址空间内;
例:映射缓存区视图

void* m_pBaseMapFile=MapViewOfFile(m_hMapFile,FILE_MAP_READ|FILE_MAP_WRITE,0,0,0);

3、客户端进程访问共享内存对象,需要通过内存对象名调用OpenFileMapping函数,以获得共享内存对象的句柄

HANDLE m_hMapFile =OpenFileMapping(FILE_MAP_WRITE,FALSE," zzj");

4、如果客户端进程获得共享内存对象的句柄成功,则调用MapViewOfFile函数来映射对象视图。用户可以使用该对象视图来进行数据读写操作,以达到数据通讯的目的。
例:映射缓存区视图

void* m_pBaseMapFile=MapViewOfFile(m_hMapFile,FILE_MAP_READ|FILE_MAP_WRITE,0,0,0);

5、当用户进程结束使用共享内存后,调用UnmapViewOfFile函数以取消其地址空间内的视图:

if (m_pBaseMapFile)
{
  UnmapViewOfFile(m_pBaseMapFile);
  SharedMapView=NULL;
}

三、使用文件映射实现共享内存。

FileMapping用于将存在于磁盘的文件放进一个进程的虚拟地址空间,并在该进程的虚拟地址空间中产生一个区域用于“存放”该文件,这个空间就叫做File View(存放在进程的虚拟内存中),系统并同时产生一个File Mapping Object(存放于物理内存中)用于维持这种映射关系,这样当多个进程需要读写那个文件的数据时,它们的File View其实对应的都是同一个File Mapping Object,这样做可节省内存和保持数据的同步性,并达到数据共享的目的。

当然在一个应用向文件中写入数据时,其它进程不应该去读取这个正在写入的数据。这就需要进行一些同步的操作。下边来看一下具体的API。

CreateFileMaping的用法:

HANDLE CreateFileMapping( //返回FileMapping Object的句柄
HANDLE hFile, //想要产生映射的文件的句柄
LPSECURITY_ATTRIBUTES lpAttributes, //安全属性(只对NT和2000生效)
DWORD flProtect, //保护标致
DWORD dwMaximumSizeHigh, //在DWORD的高位中存放
File Mapping Object //的大小
DWORD dwMaximumSizeLow, //在DWORD的低位中存放
File Mapping Object //的大小(通常这两个参数有一个为0)
LPCTSTR lpName //File Mapping Object的名称。
);

1)物理文件句柄

任何可以获得的物理文件句柄,如果你需要创建一个物理文件无关的内存映射也无妨,将它设置成为 0xFFFFFFFF(INVALID_HANDLE_VALUE)就可以了.

如果需要和物理文件关联,要确保你的物理文件创建的时候的访问模式和"保护设置"匹配,比如:物理文件只读,内存映射需要读写就会发生错误。推荐你的物理文件使用独占方式创建。

如果使用 INVALID_HANDLE_VALUE,也需要设置需要申请的内存空间的大小,无论物理文件句柄参数是否有效,这样 CreateFileMapping就可以创建一个和物理文件大小无关的内存空间给你,甚至超过实际文件大小,如果你的物理文件有效,而大小参数为0,则返回给你的是一个和物理文件大小一样的内存空间地址范围。返回给你的文件映射地址空间是可以通过复制,集成或者命名得到,初始内容为0。

2)保护设置

就是安全设置,不过一般设置NULL就可以了,使用默认的安全配置. 在win2k下如果需要进行限制,这是针对那些将内存文件映射共享给整个网络上面的应用进程使用是,可以考虑进行限制.

3)高位文件大小

32位地址空间,设置为0。

4) 共享内存名称

命名可以包含 "Global"或者 "Local" 前缀在全局或者会话名空间初级文件映射.其他部分可以包含任何除了()以外的字符,可以参考 Kernel Object Name Spaces.

5)调用CreateFileMapping的时候GetLastError的对应错误

ERROR_FILE_INVALID 如果企图创建一个零长度的文件映射,应有此报
ERROR_INVALID_HANDLE 如果发现你的命名内存空间和现有的内存映射,互斥量,信号量,临界区同名就麻烦了
ERROR_ALREADY_EXISTS 表示内存空间命名已经存在

使用函数CreateFileMapping创建一个想共享的文件数据句柄,然后使用MapViewOfFile来获取共享的内存地址,然后使用OpenFileMapping函数在另一个进程里打开共享文件的名称,这样就可以实现不同的进程共享数据。

代码示例:这个程序包括一个客户端和一个服务端,服务端创建共享内存,客户端打开共享内存,两者通过两个事件互斥访问共享内存,实现一个小功能,就是服务端进程从控制台读入数据发送给客户端进程。

服务端:

#include "stdafx.h"
#include <Windows.h>
#include <iostream>
using namespace std; 

int main()
{
  HANDLE hMutex      = NULL;
  HANDLE hFileMapping   = NULL;
  LPVOID lpShareMemory  = NULL;
  HANDLE hServerWriteOver = NULL;
  HANDLE hClientReadOver = NULL; 

  //create share memory
  hFileMapping = CreateFileMapping(INVALID_HANDLE_VALUE,
    NULL,
    PAGE_READWRITE,
    0,
    1024*1024,
    L"ShareMemoryTest");
  if (NULL == hFileMapping)
  {
    cout << "CreateFileMapping fail:" << GetLastError() << endl;
    goto SERVER_SHARE_MEMORY_END;
  } 

  lpShareMemory = MapViewOfFile(hFileMapping,
    FILE_MAP_ALL_ACCESS,
    0,
    0,   //memory start address
    0);   //all memory space
  if (NULL == lpShareMemory)
  {
    cout << "MapViewOfFile" << GetLastError() << endl;
    goto SERVER_SHARE_MEMORY_END;
  } 

  hMutex = CreateMutex(NULL, FALSE, L"SM_Mutex");
  if (NULL == hMutex || ERROR_ALREADY_EXISTS == GetLastError())
  {
    cout << "CreateMutex" << GetLastError() << endl;
    goto SERVER_SHARE_MEMORY_END;
  }//多个线程互斥访问 

  //send data
  hServerWriteOver = CreateEvent(NULL,
    TRUE,
    FALSE,
    L"ServerWriteOver");
  hClientReadOver = CreateEvent(NULL,
    TRUE,
    FALSE,
    L"ClientReadOver");
  if (NULL == hServerWriteOver ||
    NULL == hClientReadOver)
  {
    cout << "CreateEvent" << GetLastError() << endl;
    goto SERVER_SHARE_MEMORY_END;
  } 

  char p = 0;
  char* q = (char*)lpShareMemory;
  do
  {
    p = getchar();
    if (WaitForSingleObject(hClientReadOver, 5*1000) != WAIT_OBJECT_0)
      goto SERVER_SHARE_MEMORY_END;
    q[0] = p;
    if (!ResetEvent(hClientReadOver)) goto SERVER_SHARE_MEMORY_END;//把指定的事件对象设置为无信号状态
    if (!SetEvent(hServerWriteOver)) goto SERVER_SHARE_MEMORY_END;//把指定的事件对象设置为有信号状态
  } while (p != '\n'); 

SERVER_SHARE_MEMORY_END:
  //release share memory
  if (NULL != hServerWriteOver)  CloseHandle(hServerWriteOver);
  if (NULL != hClientReadOver)  CloseHandle(hClientReadOver);
  if (NULL != lpShareMemory)   UnmapViewOfFile(lpShareMemory);
  if (NULL != hFileMapping)    CloseHandle(hFileMapping);
  if (NULL != hMutex)       ReleaseMutex(hMutex);
  return 0;
}

客户端:

#include "stdafx.h"
#include <Windows.h>
#include <iostream>
using namespace std;
int main()
{
  HANDLE hMutex      = NULL;
  HANDLE hFileMapping   = NULL;
  LPVOID lpShareMemory  = NULL;
  HANDLE hServerWriteOver = NULL;
  HANDLE hClientReadOver = NULL; 

  hMutex = OpenMutex(MUTEX_ALL_ACCESS,
    FALSE,
    L"SM_Mutex");
  if (NULL == hMutex)
  {
    if (ERROR_FILE_NOT_FOUND == GetLastError())
    {
      cout << "OpenMutex fail: file not found!" << endl;
      goto CLIENT_SHARE_MEMORY_END;
    }
    else
    {
      cout << "OpenMutex fail:" << GetLastError() << endl;
      goto CLIENT_SHARE_MEMORY_END;
    }
  } 

  if (WaitForSingleObject(hMutex, 5000) != WAIT_OBJECT_0)//hMutex 一旦互斥对象处于有信号状态,则该函数返回
  {
    DWORD dwErr = GetLastError();
    goto CLIENT_SHARE_MEMORY_END;
  } 

  //open share memory
  hFileMapping = OpenFileMapping(FILE_MAP_ALL_ACCESS,
    FALSE,
    L"ShareMemoryTest");
  if (NULL == hFileMapping)
  {
    cout << "OpenFileMapping" << GetLastError() << endl;
    goto CLIENT_SHARE_MEMORY_END;
  } 

  lpShareMemory = MapViewOfFile(hFileMapping,
    FILE_MAP_ALL_ACCESS,
    0,
    0,
    0);
  if (NULL == lpShareMemory)
  {
    cout << "MapViewOfFile" << GetLastError() << endl;
    goto CLIENT_SHARE_MEMORY_END;
  } 

  //read and write data
  hServerWriteOver = CreateEvent(NULL,
    TRUE,
    FALSE,
    L"ServerWriteOver");
  hClientReadOver = CreateEvent(NULL,
    TRUE,
    FALSE,
    L"ClientReadOver");
  if (NULL == hServerWriteOver ||
    NULL == hClientReadOver)
  {
    cout << "CreateEvent" << GetLastError() << endl;
    goto CLIENT_SHARE_MEMORY_END;
  } 

  char p = 0;
  char* q = (char*)lpShareMemory;
  do
  {
    if (!SetEvent(hClientReadOver))
      goto CLIENT_SHARE_MEMORY_END; 

    if (WaitForSingleObject(hServerWriteOver, INFINITE) != WAIT_OBJECT_0)
      goto CLIENT_SHARE_MEMORY_END;  

    p = q[0];
    putchar(p);
    if (!ResetEvent(hServerWriteOver))
      goto CLIENT_SHARE_MEMORY_END;
  } while (p != '\n'); 

CLIENT_SHARE_MEMORY_END:
  //release share memory
  if (NULL != hServerWriteOver)  CloseHandle(hServerWriteOver);
  if (NULL != hClientReadOver)  CloseHandle(hClientReadOver);
  if (NULL != lpShareMemory)   UnmapViewOfFile(lpShareMemory);
  if (NULL != hFileMapping)    CloseHandle(hFileMapping);
  if (NULL != hMutex)       ReleaseMutex(hMutex);
  return 0;
}
(0)

相关推荐

  • 实例分析一个简单的Win32程序

    本文较为详细的分析了一个Win32程序的组成.结构.实现方法及运行原理,对于进行Windows程序设计有很好的借鉴参考价值.分享给大家供大家参考之用.具体分析如下: 一.Windows程序与普通C或C++程序的不同 学过C或C++等语言的人都知道,我们写的程序都一个入口,main函数,但是在Win32程序里,我们的入口函数又是什么呢?它是怎么样运行的,跟我们用C或C++写的控制台程序又有什么不同呢? 我们先说Win32程序跟我们控制台的程序的一个很重要的不同点,就是Win32程序是一个消息响应程

  • win32使用openfilename浏览文件窗口示例

    代码如下: 复制代码 代码如下: OPENFILENAME ofn;WCHAR* szFile = new WCHAR[512];WCHAR* szFileTitle = new WCHAR[512];memset(&ofn, 0, sizeof(ofn));memset(szFile, 0, sizeof(WCHAR)*512);memset(szFileTitle, 0, sizeof(WCHAR)*512); ofn.lStructSize = sizeof(ofn);ofn.hwndOw

  • 一个win32窗口创建示例

    复制代码 代码如下: /*一个简单的win32窗口调用*/#include<Windows.h>#include<tchar.h> //声明窗口函数LRESULT CALLBACK WindowProc(HWND hwnd,       UINT uMsg,       WPARAM wParam,       LPARAM lparam       );int WINAPI WinMain(     HINSTANCE hInstance,     HINSTANCE hPrev

  • WIN32程序获取父进程ID的方法

    学过windows程序设计的人都知道,windows中的进程是可以有父子关系的,拥有父子关系的进程,一旦父进程结束,子进程有会随之退出.但是如果进程之间没有父子关系,我们如何让子进程在父进程退出是也同时跟着退出呢?方法有很多,本文介绍其中的一种利用父进程ID的方案,实现的原理很简单:先获取父进程的ID,然后通过ID来获取父进程Handle,通过监视父进程的Handle来决定子进程是否退出.所以,这里的关键就是如何获取父进程的ID. 为了获取父进程ID,这里需要用到一个微软未公开的API: NTS

  • win32 api实现简单的消息窗口示例

    复制代码 代码如下: #include<windows.h> LRESULT CALLBACK WndProc(HWND,UINT,WPARAM,LPARAM); int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,PSTR szCmdLine,int iCmdShow){static TCHAR szAppName[]=TEXT("HelloWin");HWND hwnd;MSG msg;WNDCL

  • Win32应用程序(SDK)设计原理详解

    一般来说所谓的Win32应用程序开发,就是在C语言的层面上,直接使用Win32 API(Application Programming Interface:系统开放出来,给程序员使用的接口.)来开发Windows应用程序或者系统程序.虽然现在直接用Win32 API开发应用程序的人已经不多了,但是深入理解Windows系统程序设计原理,仍然是成为Windows开发高手的必经之路. 所谓的Win32,其实是一个API规范,与UNIX系统编程接口标准POSIX是相对应的.下面是进行直接的WIN32

  • tc编译的dos程序和vc编译的win32控制台程序的异同

    它们有着本质的区别:1:一个是16的DOS程序,一个是标准的32的Windows应用程序.前者依然受着64KB内存分段约束,后者无忧无虑地使用着4GB内存地址空间.它们本来就是运行在不同的操作系统上的.别把他们混淆.DOS程序可以运行在windows上是因为有NTVDM.而你把VC生成的控制台程序放到DOS下只会得到!?This program cannot be run in DOS mode.2:如果你是用标准库编写的话,代码应该没多大差异的.但执行时调用库方式却差天共地.前者通过API模拟

  • 终于明白了tc编译的dos程序和vc编译的win32控制台程序的区别

    它们有着本质的区别:1:一个是16的DOS程序,一个是标准的32的Windows应用程序.前者依然受着64KB内存分段约束,后者无忧无虑地使用着4GB内存地址空间.它们本来就是运行在不同的操作系统上的.别把他们混淆.DOS程序可以运行在windows上是因为有NTVDM.而你把VC生成的控制台程序放到DOS下只会得到!?This program cannot be run in DOS mode.2:如果你是用标准库编写的话,代码应该没多大差异的.但执行时调用库方式却差天共地.前者通过API模拟

  • Win32 下病毒设计入门详细说明

    本文假定你对dos下的病毒和386PM有一定的了解. 1.感染任何一个病毒都需要有寄主,把病毒代码加入寄主程序中(伴侣病毒除外) 以下说明如何将病毒代码嵌入PE文件中,有关PE文件的结构请看以前的文章. PE文件的典型结构:MZ Header DOS STUB CODE PE HEADER OPTIONAL HEADER SECTION TABLE SECTION 1 SECTION 2 ... IMPORT TABLE EXPORT TABLE 和DOS的可执行文件类似,PE的代码映象分为几个

  • win32 api实现2048游戏示例

    自学的win32编程,写了一个win32 API版2048,自己摸索着写的,按上下左右箭头开始游戏 复制代码 代码如下: #include <windows.h>#include <stdlib.h>#include <stdio.h>#include <ctime>LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);//全局操作数组int arr[4][4] = { { 0, 0, 0, 0 }, {

随机推荐