Windows线程同步(下)
- 线程同步三:事件
CreateEvent:Creates or opens a named or unnamed event object.
HANDLE WINAPI CreateEvent(
_In_opt_ LPSECURITY_ATTRIBUTES lpEventAttributes,
_In_ BOOL bManualReset,
_In_ BOOL bInitialState,
_In_opt_ LPCTSTR lpName
);
函数说明:
第一个参数表示安全控制,一般直接传入NULL。
第二个参数确定事件是手动置位还是自动置位,传入TRUE表示手动置位,传入FALSE表示自动置位。如果为自动置位,则对该事件调用WaitForSingleObject()后会自动调用ResetEvent()使事件变成未触发状态。
第三个参数表示事件的初始状态,传入TRUR表示已触发。
第四个参数表示事件的名称,传入NULL表示匿名事件。
OpenEvent:Opens an existing named event object.
HANDLE WINAPI OpenEvent(
_In_ DWORD dwDesiredAccess,
_In_ BOOL bInheritHandle,
_In_ LPCTSTR lpName
);
函数说明:
第一个参数表示访问权限,对事件一般传入EVENT_ALL_ACCESS。详细解释可以查看MSDN文档。
第二个参数表示事件句柄继承性,一般传入TRUE即可。
第三个参数表示名称,不同进程中的各线程可以通过名称来确保它们访问同一个事件。
SetEvent:Sets the specified event object to the signaled state.
BOOL WINAPI SetEvent(
_In_ HANDLE hEvent
);
函数功能:触发事件
函数说明:每次触发后,必有一个或多个处于等待状态下的线程变成可调度状态。
ResetEvent:Sets the specified event object to the nonsignaled state.
BOOL WINAPI ResetEvent(
_In_ HANDLE hEvent
);
函数功能:将事件设为末触发
CloseHandle:事件销毁
举例:与Mutex中的例子相比,这里只修改了两行代码
#include <iostream> #include <windows.h> using namespace std; long g_nNum; #define MAX_THREADS 20 HANDLE g_hThreadEvent; CRITICAL_SECTION g_csThreadCode; DWORD WINAPI MyThreadFunction(LPVOID lpParam) { int nThreadNum = *(int *)lpParam; SetEvent(g_hThreadEvent);//释放互斥量以便供其他线程获取 Sleep(500); EnterCriticalSection(&g_csThreadCode); g_nNum++; Sleep(500); cout << "参数 " << nThreadNum << " : " << "资源:" << g_nNum << endl; LeaveCriticalSection(&g_csThreadCode); return 0; } int main() { g_hThreadEvent = CreateEvent(NULL, FALSE, TRUE, NULL); InitializeCriticalSection(&g_csThreadCode); HANDLE hThreadArray[MAX_THREADS]; DWORD dwThreadIdArray[MAX_THREADS] = { 0 }; g_nNum = 0; int i = 0; while (i < MAX_THREADS) { WaitForSingleObject(g_hThreadEvent, INFINITE);//等待获取互斥量 hThreadArray[i] = (HANDLE)CreateThread(NULL, 0, MyThreadFunction, &i, 0, &dwThreadIdArray[i]); i++; } WaitForMultipleObjects(MAX_THREADS, hThreadArray, TRUE, INFINITE); //销毁互斥量和关键段 CloseHandle(g_hThreadEvent); DeleteCriticalSection(&g_csThreadCode); for (i = 0; i < MAX_THREADS; i++) CloseHandle(hThreadArray[i]); system("pause"); return 0; }
模仿CRITICAL_SECTION例子中的方式,在WaitForSingleObject和SetEvent上分别下断点:
然后这次再运行,与CRITICAL_SECTION中例子不同的现象发生了,这次会轮流在WaitForSingleObject和SetEvent处断下,而运行结果也显示:
参数不再有重复的了,说明利用Event实现了主线程和各个子线程间的同步。
但是为什么Event可以?
Mutex有一个“所有者”的概念,只有拥有互斥对象的线程才具有访问资源 的权限,由于互斥对象只有一个,因此就决定了任何情况下此共享资源都不会同时被多个线程所访问。当前占据资源的线程在任务处理完后应将拥有的互斥对象交出,以便其他线程在获得后得以访问资源。事件对象也可以通过通知操作的方式来保持线程的同步。可以这么理解,对于Mutex,A线程占有资源S,A可以重复多次访问资源S,除非A自身决定释放,否则其它线程无法去访问S。但是对于Event就不同了。A线程Wait将事件变成未触发状态进行它的操作,操作后,B线程可以触发这个事件,继续B的后续的操作,若无线程释放,则A也无法重复的去获取这个事件以重复进行操作。
此外,Event还有一个API:PulseEvent
BOOL WINAPI PulseEvent(
_In_ HANDLE hEvent
);
函数功能:将事件触发后立即将事件设置为未触发,相当于触发一个事件脉冲。
函数说明:这是一个不常用的事件函数,此函数相当于SetEvent()后立即调用ResetEvent();此时情况可以分为两种:
1.对于手动置位事件,所有正处于等待状态下线程都变成可调度状态。
2.对于自动置位事件,所有正处于等待状态下线程只有一个变成可调度状态。
此后事件是末触发的。该函数不稳定,因为无法预知在调用PulseEvent ()时哪些线程正处于等待状态。
举例:
设为自动置位:
#include <iostream> #include <windows.h> using namespace std; HANDLE g_hThreadEvent; //快线程 DWORD WINAPI FastThreadFun(LPVOID lpParam) { Sleep(10); //用这个来保证各线程调用等待函数的次序有一定的随机性 //cout << "启动线程" << lpParam << endl; printf("%s 启动线程\n", lpParam); WaitForSingleObject(g_hThreadEvent, INFINITE);//线程在等待时间 //自动就有个ResetEvent printf("%s 等到事件被触发 顺利结束\n", lpParam); return 0; } //慢线程 DWORD WINAPI SlowThreadFun(LPVOID lpParam) { Sleep(10); printf("%s 启动线程\n", lpParam); WaitForSingleObject(g_hThreadEvent, INFINITE); //自动就有个ResetEvent 置为未触发状态 printf("%s 等到事件被触发 顺利结束\n", lpParam); return 0; } int main() { BOOL bManualReset = FALSE;//自动置位FALSE 即自动ResetEvent(未触发状态) //第三个参数是FALSE,标明事件被初始化时,其状态是未触发的 g_hThreadEvent = CreateEvent(NULL, bManualReset, FALSE, NULL); //"数组"的数组,其实"数组"就是地址。szFastThreadName[5][30]表示5个"数组",即5个地址.szFastThreadName[i]就是一个地址 //观察内存也可发现,这个数组szFastThreadName[5],保存了5个地址,每个地址处开辟了30字节的空间 //int a[5] = { 1, 2, 3, 4, 5 };//这里每四个字节存一个数 char szFastThreadName[5][30] = { "Fast1000", "Fast1001", "Fast1002", "Fast1003", "Fast1004" };//这里每30个字节存一个字符数组 //但a[i]是第一个数字,而szFastThreadName[i]不是第i个数组,而是第i个数组的地址。开辟了5*30字节空间,但是用地址代表每个数组 //我们可以想象,a是数组a[5]的首地址;那么,szSlowThreadName[1],就是数组szSlowThreadName[1][30]的首地址 char szSlowThreadName[2][30] = { "Slow196", "Slow197" }; DWORD dwFastThreadID[5]; DWORD dwSlowThreadID[2]; int i; for (i = 0; i < 5; i++) CreateThread(NULL, 0, FastThreadFun, szFastThreadName[i], 0, &dwFastThreadID[i]); for (i = 0; i < 2; i++) CreateThread(NULL, 0, SlowThreadFun, szSlowThreadName[i], 0, &dwSlowThreadID[i]); Sleep(50); //保证快线程已经全部启动 cout << "现在主线程触发一个事件脉冲 - PulseEvent()" << endl; PulseEvent(g_hThreadEvent); //自动的话,已经有了一个ResetEvent // SetEvent(g_hThreadEvent); //触发 // ResetEvent(g_hThreadEvent); //Sleep(3000); cout << "时间到,主线程结束运行" << endl; CloseHandle(g_hThreadEvent); system("pause"); return 0; }
自动置位的思路是这样的,由于CreateEvent的第三个参数是FALSE,所以,这个事件一上来是未激活的。主线程先起来7个子线程,当主线程执行一个PulseEvent时,会在某一瞬间触发一个事件,所以会有一个正在Wait的子线程被激活而顺利执行,但由于是自动置位的,所以这个被激活的子线程在WaitForSingleObject后,就立刻自动ResetEvent了,于是其它的线程无法被激活,由于主线程没有WaitForSingleObject,所以,只有主线程继续顺利执行下去。
而如果设为手动置位结果如下:
因为,主线程PulseEvent之后,某个线程Wait到这个事件,而它在WaitForSingleObject并未ResetEvent,所以事件还是触发状态,其它的正在Wait的子线程仍可以Wait到,所以激活了全部子线程。
参考:http://blog.csdn.net/morewindows/article/details/7445233
- 线程同步四:信号量(Semaphore)
CreateSemaphore
HANDLE WINAPI CreateSemaphore(
_In_opt_ LPSECURITY_ATTRIBUTES lpSemaphoreAttributes,
_In_ LONG lInitialCount,
_In_ LONG lMaximumCount,
_In_opt_ LPCTSTR lpName
);
函数功能:创建信号量
函数说明:
第一个参数表示安全控制,一般直接传入NULL。
第二个参数表示初始资源数量。
第三个参数表示最大并发数量。
第四个参数表示信号量的名称,传入NULL表示匿名信号量。
OpenSemaphore
HANDLE WINAPI OpenSemaphore(
_In_ DWORD dwDesiredAccess,
_In_ BOOL bInheritHandle,
_In_ LPCTSTR lpName
);
函数功能:打开信号量
函数说明:
第一个参数表示访问权限,对一般传入SEMAPHORE_ALL_ACCESS。详细解释可以查看MSDN文档。
第二个参数表示信号量句柄继承性,一般传入TRUE即可。
第三个参数表示名称,不同进程中的各线程可以通过名称来确保它们访问同一个信号量。
ReleaseSemaphore
BOOL WINAPI ReleaseSemaphore(
_In_ HANDLE hSemaphore,
_In_ LONG lReleaseCount,
_Out_opt_ LPLONG lpPreviousCount
);
函数功能:递增信号量的当前资源计数
函数说明:
第一个参数是信号量的句柄。
第二个参数表示增加个数,必须大于0且不超过最大资源数量。
第三个参数可以用来传出先前的资源计数,设为NULL表示不需要传出。
注意:当前资源数量大于0,表示信号量处于触发,等于0表示资源已经耗尽故信号量处于末触发。在对信号量调用等待函数时,等待函数(WaitForSingleObject)会检查信号量的当前资源计数,如果大于0(即信号量处于触发状态),减1后返回让调用线程继续执行。一个线程可以多次调用等待函数来减小信号量。
CloseHandle
最后一个 信号量的清理与销毁.由于信号量是内核对象,因此使用CloseHandle()就可以完成清理与销毁了。
在经典多线程问题中设置一个信号量和一个关键段。用信号量处理主线程与子线程的同步,用关键段来处理各子线程间的互斥。举例:
#include <stdio.h> #include <windows.h> long g_nNum; const int THREAD_NUM = 10; //信号量与关键段 HANDLE g_hThreadParameter; CRITICAL_SECTION g_csThreadCode; DWORD WINAPI ChildThread(LPVOID lpParam) { int nThreadNum = *(int *)lpParam; ReleaseSemaphore(g_hThreadParameter, 1, NULL);//信号量++ 变为1,主线程中的WaitForSingleObject //才能Wait到信号量,从而继续的执行 Sleep(50); EnterCriticalSection(&g_csThreadCode); ++g_nNum; Sleep(0); printf("线程:%d 资源:%d\n", nThreadNum, g_nNum); LeaveCriticalSection(&g_csThreadCode); return 0; } int main() { //初始化信号量和关键段 g_hThreadParameter = CreateSemaphore(NULL, 0, 1, NULL);//初始0个资源,最大允许1个同时访问 InitializeCriticalSection(&g_csThreadCode); HANDLE handle[THREAD_NUM]; g_nNum = 0; int i = 0; while (i < THREAD_NUM) { handle[i] = (HANDLE)CreateThread(NULL, 0, ChildThread, &i, 0, NULL); WaitForSingleObject(g_hThreadParameter, INFINITE);//等待信号量>0 ++i; } WaitForMultipleObjects(THREAD_NUM, handle, TRUE, INFINITE); DeleteCriticalSection(&g_csThreadCode); CloseHandle(g_hThreadParameter); for (i = 0; i < THREAD_NUM; i++) CloseHandle(handle[i]); system("pause"); return 0; }
CreateSemaphore的第二个参数位0,第三个参数为1,表示信号量初始为0,最多允许1个线程同时访问。主线程中CreateThread之后,信号量个数为0,所以主线程只能一直Wait,直到子线程中有一个ReleaseSemaphore是的信号量个数加1大于0,
这时主线程才能Wait到信号量,且自动减1。而后边主线程如果再想执行,也必须要等到其它子线程先ReleaseSemaphore才行。
可以看出来,信号量也可以解决线程之间的同步问题。
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
· Sdcb Chats 技术博客:数据库 ID 选型的曲折之路 - 从 Guid 到自增 ID,再到
· 语音处理 开源项目 EchoSharp
· 《HelloGitHub》第 106 期
· Spring AI + Ollama 实现 deepseek-r1 的API服务和调用
· 使用 Dify + LLM 构建精确任务处理应用