原子访问:是指线程在访问资源时能够确保所有其他线程都不在同一时间内访问相同的资源。
对于互锁函数,需要了解它们运行的速度极快。调用一个互锁函数,通常会导致执行几个CPU周期(通常小于50),并且不会从用户方式转为内核方式(通常这个需要执行1000个CPU周期)。
几个互锁函数:
(1) LONG InterlockedExchangeAdd ( LPLONG Addend, LONG Increment );
Addend为长整型变量的地址,Increment为想要在Addend指向的长整型变量上增加的数值(可以是负数)。这个函数的主要作用是保证这个加操作为一个原子访问。
(2) LONG InterlockedExchange( LPLONG Target, LONG Value );
用第二个参数的值取代第一个参数指向的值。函数返回值为原始值。
(3) PVOID InterlockedExchangePointer( PVOID *Target, PVOID Value );
用第二个参数的值取代第一个参数指向的值。函数返回值为原始值。
(4) LONG InterlockedCompareExchange(LPLONG Destination, LONG Exchange, LONG Comperand );
如果第三个参数与第一个参数指向的值相同,那么用第二个参数取代第一个参数指向的值。函数返回值为原始值。
(5) PVOID InterlockedCompareExchangePointer (PVOID *Destination, PVOID Exchange, PVOID Comperand );
如果第三个参数与第一个参数指向的值相同,那么用第二个参数取代第一个参数指向的值。函数返回值为原始值。
有些变量是用volatile关键字声明的。当两个线程都要用到某一个变量且该变量的值会被改变时,应该用volatile声明,该关键字的作用是防止优化编译器把变量从内存装入CPU寄存器中。如果变量被装入寄存器,那么两个线程有可能一个使用内存中的变量,一个使用寄存器中的变量,这会造成程序的错误执行。volatile的意思是让编译器每次操作该变量时一定要从内存中真正取出,而不是使用已经存在寄存器中的值。
SRWLock的目的和关键段相同,都是保护一个资源,不让其他线程访问。SRWLock允许我们区分对待读线程和写线程;多个读线程可以共享访问资源,写线程独占资源。使用完成后不需要清理工作。
相关函数:InitializeSWRLock AcquireSRWLockShared,ReleaseSRWLockShared(读的函数)
AcquireSRWLockExclusive,ReleaseSRWLockExclusive(写的函数)
性能比较:如果希望在应用程序中得到最佳性能,那么首先应该尝试不要共享数据,然后依次使用volatile读取,volatile写入,Interlocked API,SRWLock,关键段。当且仅当所有这些都不能满足要求的时候,再使用内核对象。
条件变量:为了以原子方式把锁释放并将自己阻塞,直到某一条件成立为止。就可以使用条件变量,分为两组,一组配合关键段,一组配合slim锁
睡眠函数:SleepConditionVariableCS,SleepConditionVariableSRW
唤醒函数:WakeConditionVariable,WakeAllConditionVariable
窍门和技巧:1、以原子方式操作一组对象时使用一个锁;
2、同时访问多个逻辑资源,当需要获得多个锁时,必须在代码任何地方以完全相同的顺序来获得资源的锁;另外,调用LeaveCriticalSection的时候顺序无关紧要,因为该函数从来不会让线程进入等待;
3、不要长时间占用锁如果仅仅是获取变量的值再用于其他操作,仅需要获取变量值时加锁,把变量的值存在一个临时变量中,然后释放锁,最后用临时变量去做其他的事,这样可以防止一些不必要的函数被包含在锁中,造成长时间占用锁。