首页 > 代码库 > 线程同步——用户模式下线程同步——Slim读写锁实现线程同步

线程同步——用户模式下线程同步——Slim读写锁实现线程同步

  1 //Slim读/写锁实现线程同步  2 SRWlock 的目的和关键段相同:对同一资源进行保护,不让其它线程访问。  3 但是,与关键段不同的是,SRWlock允许我们区分哪些想要读取资源的线程(读取者线程)  4 和哪些想要更新资源值的线程(写入者线程)。让所有读取者资源在同一时刻访问共享资源应该是  5 可行的,这是因为仅仅读取资源并不存在破坏数据的风险。只有当写入者线程想要对资源进行更新时才需要同步。  6 这种情况下,写入者线程应该独占资源访问权:任何线程,无论是读取还是写入者线程,都不许访问资源。  7 这就是SRWlock强大之处。  8   9 //使用步骤: 10 //(1)必须先定义 CRITICAL_SECTION 结构 11 SRWLOCK g_cs; 12 //(2)初始化关键段 SRWLOCK 13 InitializeSRWLock(&g_cs); 14 //3)写入者线程 15 DWORD WINAPI ThreadFunOne(PVOID pvParam)  16 { 17     AcquireSRWLockExclusive(&g_cs)  ; 18  19     //写入资源  应该放在AcquireSRWLockExclusive和ReleaseSRWLockExclusive函数之间 20     g_x ++ ; 21  22     ReleaseSRWLockExclusive(&g_cs); 23     return 0; 24 } 25 //4)读取者线程 26 DWORD WINAPI ThreadFunTwo(PVOID pvParam) 27 { 28     AcquireSRWLockShared(&g_cs)  ; 29  30     //读取资  源应该放在AcquireSRWLockShared和ReleaseSRWLockShared函数之间 31     cout<<"ThreadFunTwo:"<<g_x<<endl  ; 32  33     ReleaseSRWLockShared(&g_cs); 34     return 0; 35 } 36  37 不存在用来删除或者销毁 SRWLOCK 的函数,因为系统会自动执行清理工作 38  39 与关键段相比,SRWlock 缺乏两个特性。 40 1)不存在TryEnter(Shared/Exclusive)SRWLock之类的函数:如果锁已经被占用, 41 那么调用AcquireSRWLock(Shared/Exclusive)会阻塞调用线程。 42 2)不能递归的获得 SRWLOCK 也就是说一个线程不能多次写入资源而多次锁定资源, 43 然后多次调用ReleaseSRWLock*来释放资源锁定。 44  45  46  47  48  49 #include "windows.h" 50 #include "iostream" 51 using namespace std; 52 long g_x = 0 ; 53 //(1)必须先定义 CRITICAL_SECTION 结构 54 SRWLOCK g_cs; 55  56 //定义线程函数1 57 DWORD WINAPI ThreadFunOne(PVOID pvParam) ; 58  59 //定义线程函数2 60 DWORD WINAPI ThreadFunTwo(PVOID pvParam); 61  62 int main() 63 { 64  65     //(2)初始化关键段 SRWLOCK 66     InitializeSRWLock(&g_cs); 67  68     //创建线程1 69     HANDLE hThreadOne = CreateThread(NULL,0,ThreadFunOne,0,0,NULL); 70     CloseHandle(hThreadOne); 71  72     //创建线程2 73     HANDLE hThreadTwo = CreateThread(NULL,0,ThreadFunTwo,0,0,NULL); 74     CloseHandle(hThreadTwo); 75  76     //让主线程先挂起,确保其它线程执行完成 77     Sleep(1000);  78     cout<<g_x<<endl; 79  80     //(4)清理CRITICAL_SECTION结构,必须确保已经没有资源使用此关键段,否则会出现不可预料的结果 81      82     return 0 ; 83 } 84  85 DWORD WINAPI ThreadFunOne(PVOID pvParam)  86 { 87     AcquireSRWLockExclusive(&g_cs)  ; 88  89     //写入资源  应该放在AcquireSRWLockExclusive和ReleaseSRWLockExclusive函数之间 90     g_x ++ ; 91  92     ReleaseSRWLockExclusive(&g_cs); 93     return 0; 94 } 95  96 DWORD WINAPI ThreadFunTwo(PVOID pvParam) 97 { 98     AcquireSRWLockShared(&g_cs)  ; 99 100     //读取资  源应该放在AcquireSRWLockShared和ReleaseSRWLockShared函数之间101      cout<<"ThreadFunTwo:"<<g_x<<endl  ;102 103     ReleaseSRWLockShared(&g_cs);104     return 0;105 }106