Windows线程同步【3】互斥锁(Mutex)

我们前面讲过的临界区,如同一个小房间,张三进去了,李四就不能进,如果李四要进,必须等张三出来。

今天我们要讲的互斥锁,像一个物件,这个物件只能同时被一个线程持有。如此一来,便可以通过互斥锁来实现线程的同步。


一、创建

创建互斥锁的方法是调用函数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函数的第三个参数传入一个字符串,那么所创建的锁就是命名的。当一个命名的锁被创建出来以后,当前进程和其他进程如果试图创建相同名字的锁,CreateMutex会返回原来那把锁的句柄,并且GetLastError函数会返回ERROR_ALREADY_EXISTS。这个特点可以使一个程序在同一时刻最多运行一个实例。

如下:

int __stdcall wWinMain(HINSTANCE hInstance, HINSTANCE, wchar_t *pCmdLine, int nCmdShow)
{
    HANDLE hMutex = CreateMutex(nullptr, FALSE, L"virtuanes");
    if (hMutex != nullptr && GetLastError() == ERROR_ALREADY_EXISTS) // 已经有了一个实例
    {
        return 0;
    }
    // 其他代码
}

你可能感兴趣的:(windows,线程,同步,互斥,互斥锁)