共享内存指 (shared memory)在多处理器的计算机系统中,可以被不同中央处理器(CPU)访问的大容量内存。由于多个CPU需要快速访问存储器,这样就要对存储器进行缓存(Cache)。任何一个缓存的数据被更新后,由于其他处理器也可能要存取,共享内存就需要立即更新,否则不同的处理器可能用到不同的数据。共享内存是 Unix下的多进程之间的通信方法 ,这种方法通常用于一个程序的多进程间通信,实际上多个程序间也可以通过共享内存来传递信息。
Win 中最具弹性的同步机制就属 events 对象了。 Event 对象是一种核心对象,它的唯一目的就是成为激发状态或未激发状态。这两种状态全由程序来控制,不会成为 WaitForSingleObject() 函数的副作用。
Event 对象之所以有大用途,正是因为它们的状态完全在你掌控之下。Mutexes和semaphores就不一样了,它们的状态会因为诸如WaitForSingleObject() 之类的函数调用而变化。所以, 你可以精确告诉一个event 对象做什么事,以及什么时候去做。
CreateFileMapping(映射文件句柄,安全属性,访问权限,对象大小,共享内存大小,映射文件名)
;(注意:映射文件名双方必须一致)MapViewOfFile(共享内存地址,访问权限,文件映射起始偏移的高32位,文件映射起始偏移的低32位, 映射文件的字节数.)
CreateEventW
用于俩个进程间的同步,CreateFileMapping(映射文件句柄,安全属性,访问权限,对象大小,共享内存大小,映射文件名)
;(注意:映射文件名双方必须一致)MapViewOfFile(共享内存地址,访问权限,文件映射起始偏移的高32位,文件映射起始偏移的低32位, 映射文件的字节数.)
CreateEventW
(此时的事件名称要与另一方一致)。这样在创建事件时系统就会发现这个事件已经被另一方创建过,就直接将创建好的句柄返回来。才可实现进程同步例如 现有俩个事件g_EventRead
和g_EventWrite
;
初始状态g_EventRead
和g_EventWrite
事件的信号灯都是灭的。先启动创建方,然后启动连接方时,在初始化时将创建方的EventWrite
事件的信号灯点亮,这样创建方获得数据就可以直接写入共享内存中并将g_EventRead
灯点亮将g_EventWrite
灯弄灭。然后连接方收到数据就可以读取共享内存中的数据,再将g_EventRead
灯弄灭将g_EventWrite
灯点亮。这样就实现了进程间的通讯。
创建共享内存
HANDLE WINAPI CreateFileMapping(
_In_HANDLE hFile,
_In_opt_LPSECURITY_ATTRIBUTES lpAttributes,
_In_DWORD flProtect,
_In_DWORD dwMaximumSizeHigh,
_In_DWORD dwMaximumSizeLow,
_In_opt_LPCTSTR lpName);
参数:
将共享内存映射到本地进程
LPVOID WINAPI MapViewOfFile(
__in HANDLE hFileMappingObject,
__in DWORD dwDesiredAccess,
__in DWORD dwFileOffsetHigh,
__in DWORD dwFileOffsetLow,
__in SIZE_T dwNumberOfBytesToMap
);
参数:
创建事件
HANDLE CreateEvent(
LPSECURITY_ATTRIBUTESlpEventAttributes,// 安全属性
BOOLbManualReset,// 复位方式
BOOLbInitialState,// 初始状态
LPCTSTRlpName // 对象名称
);
参数:
1. 安全属性 默认为NULL
2. 复位方式 (true-手工恢复, false-自动恢复)
3. 初始状态 (true-有信号,false-无信号)
4. 事件名称
返回值:
在指定时间内等待事件的信号状态
DWORD WaitForSingleObject(
HANDLE hHandle,
DWORD dwMilliseconds
);
参数:
创建一个线程
HANDLE CreateThread(
LPSECURITY_ATTRIBUTES lpThreadAttributes,//SD
SIZE_T dwStackSize,//initialstacksize
LPTHREAD_START_ROUTINE lpStartAddress,//threadfunction
LPVOID lpParameter,//threadargument
DWORD dwCreationFlags,//creationoption
LPDWORD lpThreadId//threadidentifier
)
参数:
打开一个现成的文件映射对象的函数
HANDLE WINAPI OpenFileMapping(
_In_ DWORD dwDesiredAccess,
_In_ BOOL bInheritHandle,
_In_ LPCWSTR lpName
)
参数:
#include
#include
using namespace std;
#define BUF_SIZE 4096
HANDLE g_EventRead; // 读信号灯
HANDLE g_EventWrite; // 写信号灯
// 定义共享数据
char szBuffer[] = "LinXi07";
/* 读取con1串口的线程 */
DWORD __stdcall WriteThread(const LPVOID lp)
{
while (true)
{
WaitForSingleObject(g_EventWrite, INFINITE); // 等待读数据的信号
// 将数据拷贝到共享内存
strcpy((char*)lp, szBuffer);
cout << "服务发送成功!等待客户端接受:" << (char*)lp << endl;
Sleep(1000);
SetEvent(g_EventRead);
ResetEvent(g_EventWrite);
}
return DWORD();
}
int main()
{
// 创建共享文件句柄
HANDLE hMapFile = CreateFileMapping(
INVALID_HANDLE_VALUE, // 物理文件句柄 NVALID_HANDLE_VALUE 则创建一个进程间共享的对象
NULL, // 默认安全级别
PAGE_READWRITE, // 可读可写
0, // 高位文件大小
BUF_SIZE, // 低位文件大小
L"ShareMemoryPDU" // 映射文件名,即共享内存的名称
);
if (0 == hMapFile)
{
return 0;
}
// 映射缓存区视图 , 得到指向共享内存的指针
// 将hFileMapping共享内存衍射到本进程的地址空间中
LPVOID lpBase = MapViewOfFile(
hMapFile, // 共享内存的句柄
FILE_MAP_ALL_ACCESS, // 可读写许可
0,
0,
BUF_SIZE
);
if (0 == lpBase)
{
return 0;
}
g_EventRead = CreateEventW(NULL, TRUE, FALSE, TEXT("EventRead"));
if (nullptr == g_EventRead)
{
return 0;
}
g_EventWrite = CreateEventW(NULL, TRUE, TRUE, TEXT("EventWrite"));
if (nullptr == g_EventRead)
{
return 0;
}
HANDLE handle = CreateThread(NULL, 0, WriteThread, lpBase, 0, NULL);
WaitForSingleObject(handle, INFINITE);
// 解除文件映射
UnmapViewOfFile(lpBase);
// 关闭内存映射文件对象句柄
CloseHandle(hMapFile);
return 0;
}
#include
#include
#include
using namespace std;
#define BUF_SIZE 4096
HANDLE g_EventRead; // 读信号灯
HANDLE g_EventWrite; // 写信号灯
DWORD __stdcall ReadThread(const LPVOID lp)
{
while (true)
{
WaitForSingleObject(g_EventRead, INFINITE); // 等待读数据的信号
// 将数据拷贝到共享内存
// 将共享内存数据拷贝出来
char szBuffer[BUF_SIZE]{ 0 };
strcpy_s(szBuffer, (char*)lp);
std::cout << "客户数据读取成功!:" << szBuffer << endl;
ResetEvent(g_EventRead); /* 将读取信号关闭 */
SetEvent(g_EventWrite);
}
}
int main()
{
// 打开共享的文件对象
HANDLE hMapFile = OpenFileMapping(FILE_MAP_ALL_ACCESS, NULL, L"ShareMemoryPDU");
if (0 == hMapFile)
{
// 打开共享内存句柄失败
std::cout << "打开共享失败!" << endl;
return 0;
}
LPVOID lpBase = MapViewOfFile(hMapFile, FILE_MAP_ALL_ACCESS, 0, 0, 0);
if (0 == lpBase)
{
return 0;
}
g_EventRead = CreateEventW(NULL, TRUE, FALSE, TEXT("EventRead"));
if (nullptr == g_EventRead)
{
return 0;
}
g_EventWrite = CreateEventW(NULL, TRUE, TRUE, TEXT("EventWrite"));
if (nullptr == g_EventRead)
{
return 0;
}
HANDLE handle = CreateThread(NULL, 0, ReadThread, lpBase, 0, NULL);
if (0 == handle)
{
return 0;
}
WaitForSingleObject(handle, INFINITE);
// 解除文件映射
UnmapViewOfFile(lpBase);
// 关闭内存映射文件对象句柄
CloseHandle(hMapFile);
return 0;
}