我们经常会遇到同一个进程中的不同线程(或者不同进程的不同线程)需要访问同一块内存块(例如同一个指针、共享内存),为了避免同时访问,可以使用互斥量(MUTEX)。
互斥量使用步骤如下:
1、 定义一个操作句柄:HANDLE m_hMutex;
2、 创建或者打开互斥量:hMutex =CreateMutex(NULL,FALSE,”TEST_MUTEX”);
可以创建有名字的互斥量,当不同进程中的线程需要访问统一内存块时,就非常有用。一个进程创建,另一个进程只要调用
hMutex =OpenMutex(MUTEX_ALL_ACCESS,FALSE, ”TEST_MUTEX”);打开即可使用同一个互斥量进行控制。
3、 等待互斥量空闲:WaitForSingleObject(m_hMutex,INFINITE);
4、 释放互斥量,让它处于空闲状态:ReleaseMutex(m_hMutex);
5、 关闭互斥量: CloseHandle(m_hMutex);
进程中止前,一定要释放互斥体,如不慎未采取这个措施,就会将这个互斥体标记为废弃,并自动释放所有权。共享这个互斥体的其他应用程序也许仍然能够用它,但会接收到一个废弃状态信息,指出上一个所有进程未能正常关闭。这种状况是否会造成影响取决于涉及到的具体应用程序(摘自百度百科)
下面上例子,例子采用的和临界区一样的例子,但是用互斥量来实现的:
#include "stdafx.h"
#include <iostream>
#include <windows.h>
#include <process.h>
DWORD WINAPI AddProc(LPVOID Lparam);
//CRITICAL_SECTION m_cs;
HANDLE m_hMutex=NULL;
int isum=0;
int main(int argc, char* argv[])
{
//InitializeCriticalSection(&m_cs);
m_hMutex=CreateMutex(NULL,FALSE,NULL); //由于在同一个进程中使用,所以可以不需要名字
HANDLE m_Thread=(HANDLE)_beginthreadex(NULL,0,(unsigned int (__stdcall *)(void *))AddProc,NULL,CREATE_SUSPENDED,NULL);
if(NULL!=m_Thread)
{
ResumeThread(m_Thread);
CloseHandle(m_Thread);
}
while(1)
{
//EnterCriticalSection(&m_cs);
WaitForSingleObject(m_hMutex,INFINITE); //等待m_hMutex处于空闲状态
isum++;
printf("%s%d\n","main isum=",isum);
if(isum>10)
{
int i=0;
//LeaveCriticalSection(&m_cs);
ReleaseMutex(m_hMutex); //释放m_hMutex让它处于空闲状态
std::cin>>i;
//DeleteCriticalSection(&m_cs);
if(m_hMutex!=NULL)
{
CloseHandle(m_hMutex);
m_hMutex=NULL;
}
return 0;
}
//LeaveCriticalSection(&m_cs);
ReleaseMutex(m_hMutex);
}
}
DWORD WINAPI AddProc(LPVOID Lparam)
{
while(1)
{
//EnterCriticalSection(&m_cs);
WaitForSingleObject(m_hMutex,INFINITE); //等待m_hMutex处于空闲状态
isum++;
printf("%s%d\n","AddProc isum=",isum);
if(isum>10)
{
//LeaveCriticalSection(&m_cs);
ReleaseMutex(m_hMutex); //释放m_hMutex让它处于空闲状态
return 0;
}
//LeaveCriticalSection(&m_cs);
ReleaseMutex(m_hMutex); //释放m_hMutex让它处于空闲状态
}
return 0;
}
结果显示:
本文介绍了在C++中如何使用互斥量(mutex)来实现多线程间的同步,防止并发访问同一资源导致的问题。通过定义、创建、等待、释放和关闭互斥量的步骤详细阐述了其工作原理,并提供了示例代码。
431

被折叠的 条评论
为什么被折叠?



