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