我们前面讲过的临界区,如同一个小房间,张三进去了,李四就不能进,如果李四要进,必须等张三出来。
今天我们要讲的互斥锁,像一个物件,这个物件只能同时被一个线程持有。如此一来,便可以通过互斥锁来实现线程的同步。
一、创建
创建互斥锁的方法是调用函数CreateMutex:
CreateMutex(&sa, bInitialOwner, szName);
第一个参数是一个指向SECURITY_ATTRIBUTES结构体的指针,一般的情况下,可以是nullptr。
第二个参数类型为BOOL,表示互斥锁创建出来后是否被当前线程持有。
第三个参数类型为字符串(const TCHAR*),是这个互斥锁的名字,如果是nullptr,则互斥锁是匿名的。
例:
HANDLE hMutex = CreateMutex(nullptr, FALSE, nullptr);
上面的代码创建了一个匿名的互斥锁,创建出来后,当前线程不持有这个互斥锁。
二、持有
WaitForSingleObject函数可以让一个线程持有互斥锁。用法:
WaitForSingleObject(hMutex, dwTimeout);
这个函数的作用比较多。这里只介绍第一个参数为互斥锁句柄时的作用。
它的作用是等待,直到一定时间之后,或者,其他线程均不持有hMutex。第二个参数是等待的时间(单位:毫秒),如果该参数为INFINITE,则该函数会一直等待下去。
三、释放
用ReleaseMutex函数可以让当前线程“放开”一个互斥锁(不持有它了),以便让其他线程可以持有它。用法
ReleaseMutex(hMutex);
四、销毁
当程序不再需要互斥锁时,要销毁它。
CloseHandle(hMutex);
五、命名互斥锁
如果CreateMutex函数的第三个参数传入一个字符串,那么所创建的锁就是命名的。当一个命名的锁被创建出来以后,当前进程和其他进程都不能再创建相同名字的锁,直到这个锁被销毁。这个特点可以使一个程序在同一时刻最多运行一个实例。
如下:
int __stdcall wWinMain(HINSTANCE, HINSTANCE, wchar_t, int) { HANDLE hMutex = CreateMutex(nullptr, FALSE, L"virtuanes"); if (hMutex == nullptr) // 已经有另一个实例在运行 { return 0; } // 其他代码 }
时间: 2024-12-31 17:33:14