C/C++使用Windows的API实现共享内存以及同步
目录
- 共享内存
- 事件-Event
- 实现思路
- 创建方(服务端)
- 连接方:
- 进程同步:
- windows的API
- CreateFileMapping
- MapViewOfFile
- CreateEvent
- WaitForSingleObject
- CreateThread
- OpenFileMapping
- 通过共享内存实现进程间的交互
- 服务端
- 客户端
- 结论
共享内存
共享内存指 (shared memory)在多处理器的计算机系统中,可以被不同中央处理器(CPU)访问的大容量内存。由于多个CPU需要快速访问存储器,这样就要对存储器进行缓存(Cache)。任何一个缓存的数据被更新后,由于其他处理器也可能要存取,共享内存就需要立即更新,否则不同的处理器可能用到不同的数据。共享内存是 Unix下的多进程之间的通信方法 ,这种方法通常用于一个程序的多进程间通信,实际上多个程序间也可以通过共享内存来传递信息。
事件-Event
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
灯点亮。这样就实现了进程间的通讯。
windows的API
CreateFileMapping
创建共享内存
HANDLE WINAPI CreateFileMapping(
_In_HANDLE hFile,
_In_opt_LPSECURITY_ATTRIBUTES lpAttributes,
_In_DWORD flProtect,
_In_DWORD dwMaximumSizeHigh,
_In_DWORD dwMaximumSizeLow,
_In_opt_LPCTSTR lpName);
参数:
- _In_HANDLE 映射文件的句柄 NVALID_HANDLE_VALUE则创建一个进程间共享的对象
- _In_opt_LPSECURITY_ATTRIBUTES 安全属性
- _In_DWORD 保护方式
- _In_DWORD 对象的大小, 32位.
- _In_DWORD 共享内存大小 字节
- _In_opt_LPCTSTR 映射文件名,即共享内存的名称
MapViewOfFile
将共享内存映射到本地进程
LPVOID WINAPI MapViewOfFile(
__in HANDLE hFileMappingObject,
__in DWORD dwDesiredAccess,
__in DWORD dwFileOffsetHigh,
__in DWORD dwFileOffsetLow,
__in SIZE_T dwNumberOfBytesToMap
);
参数:
- hFileMappingObject 为CreateFileMapping()或OpenFileMapping()返回的文件映像对象句柄。
- dwDesiredAccess 映射对象的文件数据的访问方式,而且同样要与CreateFileMapping()函数所设置的保护属性相匹配。 可取以下值:
- FILE_MAP_ALL_ACCESS 等价于CreateFileMapping的 FILE_MAP_WRITE|FILE_MAP_READ. 文件映射对象被创建时必须指定PAGE_READWRITE 选项.
- FILE_MAP_COPY 可以读取和写入文件.写入操作会导致系统为该页面创建一份副本.在调用CreateFileMapping时必须传入PAGE_WRITECOPY保护属性.
- FILE_MAP_EXECUTE 可以将文件中的数据作为代码来执行.在调用CreateFileMapping时可以传入PAGE_EXECUTE_READWRITE或PAGE_EXECUTE_READ保护属性.
- FILE_MAP_READ 可以读取文件.在调用CreateFileMapping时可以传入PAGE_READONLY或PAGE_READWRITE保护属性.
- FILE_MAP_WRITE 可以读取和写入文件.在调用CreateFileMapping时必须传入PAGE_READWRITE保护属性.
- dwFileOffsetHigh 表示文件映射起始偏移的高32位.
- dwFileOffsetLow 表示文件映射起始偏移的低32位.(64KB对齐不是必须的)
- dwNumberOfBytes 指定映射文件的字节数.
CreateEvent
创建事件
HANDLE CreateEvent(
LPSECURITY_ATTRIBUTESlpEventAttributes,// 安全属性
BOOLbManualReset,// 复位方式
BOOLbInitialState,// 初始状态
LPCTSTRlpName // 对象名称
);
参数:
1. 安全属性 默认为NULL
2. 复位方式 (true-手工恢复, false-自动恢复)
3. 初始状态 (true-有信号,false-无信号)
4. 事件名称
返回值:
- 事件的句柄
WaitForSingleObject
在指定时间内等待事件的信号状态
DWORD WaitForSingleObject(HANDLE hHandle,
DWORD dwMilliseconds
);
参数:
- hHandle 对象句柄。可以指定一系列的对象,如Event、Job、Memory resource notification、Mutex、Process、Semaphore、Thread、Waitable timer等。
- dwMilliseconds 定时时间间隔,单位为milliseconds(毫秒)
CreateThread
创建一个线程
HANDLE CreateThread(
LPSECURITY_ATTRIBUTES lpThreadAttributes,//SD
SIZE_T dwStackSize,//initialstacksize
LPTHREAD_START_ROUTINE lpStartAddress,//threadfunction
LPVOID lpParameter,//threadargument
DWORD dwCreationFlags,//creationoption
LPDWORD lpThreadId//threadidentifier
)
参数:
- lpThreadAttributes 指向SECURITY_ATTRIBUTES型态的结构的指针。在Windows 98中忽略该参数。在Windows NT中,NULL使用默认安全性,不可以被子线程继承,否则需要定义一个结构体将它的bInheritHandle成员初始化为TRUE
- dwStackSize,设置初始栈的大小,以字节为单位,如果为0,那么默认将使用与调用该函数的线程相同的栈空间大小。任何情况下,Windows根据需要动态延长堆栈的大小。
- lpStartAddress,指向线程函数的指针,形式:@函数名,函数名称没有限制,
- lpParameter:向线程函数传递的参数,是一个指向结构的指针,不需传递参数时,为NULL。
- dwCreationFlags :线程标志,可取值如下
(1)CREATE_SUSPENDED(0x00000004):创建一个挂起的线程,
(2)0:表示创建后立即激活。
(3)STACK_SIZE_PARAM_IS_A_RESERVATION(0x00010000):dwStackSize参数指定初始的保留堆栈 的大小,否则,dwStackSize指定提交的大小。 - lpThreadId:保存新线程的id。
OpenFileMapping
打开一个现成的文件映射对象的函数
HANDLE WINAPI OpenFileMapping(_In_ DWORD dwDesiredAccess,_In_ BOOL bInheritHandle,_In_ LPCWSTR lpName)
参数:
- dwDesiredAccess 访问方式
- bInheritHandle 继承标志句柄继承选项表示内核对象被子进程继承
- lpName 指定要打开的文件映射对象名称。
通过共享内存实现进程间的交互
服务端
#include <windows.h>
#include <iostream>
using namespace std;#define BUF_SIZE 4096HANDLE 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 <iostream>
#include <windows.h>
#include <string>
using namespace std;#define BUF_SIZE 4096HANDLE 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;
}
结论
单启动服务端,将只会发出一条,然后就一直处于等待状态
先启动服务端,再启动客户端。可以看到俩进程之间进行数据共享
C/C++使用Windows的API实现共享内存以及同步相关推荐
- java 内存映射文件进程间通讯_[转]Windows环境下利用“共享内存”实现进程间通信的C/C++代码---利用CreateFileMapping和MapViewOfFile...
进程间的通信方式有很多种, 上次我们说了最傻瓜的"共享外存/文件"的方法. 那么, 在本文中, 我们即将学习"共享内存"的方式实现进程间的通信, 这是IPC最快 ...
- Windows上C++使用共享内存进行进程间通讯
共享内存 (也叫内存映射文件) 主要是通过映射机制实现的 , Windows 下进程的地址空间在逻辑上是相互隔离的 , 但在物理上却是重叠的 ; 所谓的重叠是指同一块内存区域可能被多个进程同时使用 , ...
- Windows进程通信之共享内存通信(C++)
首先是概念:https://baike.baidu.com/item/%E5%85%B1%E4%BA%AB%E5%86%85%E5%AD%98/2182364?fr=aladdin 这是比较官方的解释 ...
- Windows共享内存解析
在Windows程序开发过程中,当多个进程之间需要使用同样的数据的时候我们最好的方式就是通过共享内存进行处理(比如:当A进程运行时,进行数据处理,那么此时我想知道数据是不是正确,用B监控,那么A与B之 ...
- CreateFileMapping实现的共享内存及用法
在32位的Windows系统中,每一个进程都有权访问他自己的4GB(232=4294967296)平面地址空间,没有段,没有选择符,没有near和far指针,没有near和far函数调用,也没有内存模 ...
- boost库中共享内存的使用
什么是共享内存 共享内存是最快速的进程间通信机制.操作系统在几个进程的地址空间上映射一段内存,然后这几个进程可以在不需要调用操作系统函数的情况下在那段内存上进行读/写操作.但是,在进程读写共享内存时, ...
- 进程间通信之-共享内存Shared Memory--linux内核剖析(十一)
共享内存 共享内存是进程间通信中最简单的方式之中的一个. 共享内存是系统出于多个进程之间通讯的考虑,而预留的的一块内存区. 共享内存同意两个或很多其他进程訪问同一块内存,就如同 malloc() 函数 ...
- linux如何创建共享内存,linux实现共享内存同步的四种方法
https://blog.csdn.net/sunxiaopengsun/article/details/79869115 本文主要对实现共享内存同步的四种方法进行了介绍. 共享内存是一种最为高效的进 ...
- Linux 下的进程间通信:管道、消息队列、共享文件、共享内存
Table of Contents 无名管道 命名管道 消息队列 共享文件 示例 1. 生产者程序 示例 2. 消费者程序 共享内存 示例 3. memwriter 进程的源程序 示例 4. memr ...
最新文章
- 关闭图片 pycharm_博士大佬总结的Pycharm 常用快捷键思维导图,收藏!
- 【实验】综合实验-咔咔咔还是一顿整
- 推荐五篇论文| 轻量级的Transformer; 对比学习;ResNeSt;Shortcut Learning等
- c5.0 java_机器学习-AdaBoosting及其Java实现
- c++ 标准库中 cin.ignore()
- SAP UI5 busy dialog released more often than required
- Java8 Stream详解~归约(reduce)
- 给定一个字符串str,将str中连续两个字符为a的字符替换为b(一个或连续超过多个字符a则不替换)...
- 八皇后问题(C语言)
- 补间动画android
- 孙鑫VC学习笔记:第十五讲 (三) 增加互斥条件实现线程同步
- 图像Scaler缩放因子
- 国产超级英雄逆袭好莱坞
- SQL Server的Descending Indexes降序索引
- HKUST-Aerial-Robotics /grad_traj_optimizationPublic
- Python生成英文词云图
- 三角法激光雷达测距原理
- 二手图书交易网站-图书发布图书购买评价购物车-计算机毕业设计基于javaWebSSMspringboot框架idea开发工具asp.net和PHP
- mysql编写函数_mysql函数编写
- R语言实战-读书笔记(第1 章 R语言介绍)
热门文章
- 2021数学建模国赛总结(含题目)
- Ajax异步请求的步骤
- 现在有个字符串是按照如下格式保存的:“姓名:成绩|姓名:成绩|…..”
- eChart 中 柱状图、地图、AntDesign 的 滚动条表格、highChart 的 (venn) 韦恩图
- PhotoZoom Pro 7怎么进行参数设置
- 第十六届北京师范大学程序设计竞赛决赛
- 智慧农业-数字孪生农业大棚
- 求AUC 95%置信区间
- Java入门编程练习题
- [SSD固态硬盘闪存 3] QLC 闪存给SSD主控带来了很大的难题?