线程同步——用户模式下线程同步——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  
原文地址:https://www.cnblogs.com/yfyzy/p/3916181.html