C++线程同步 CCRITICALSECTION(临界区类)附自己写的测试代码

来源:互联网 发布:阿哥阿妹-网络 编辑:程序博客网 时间:2024/06/06 12:30

当多个线程访问一个独占性共享资源时,可以使用“临界区”对象。任一时刻只有一个线程可以拥有临界区对象,拥有临界区的线程可以访问被保护起来的资源或代码段,其他希望进入临界区的线程将被挂起等待,直到拥有临界区的线程放弃临界区时为止,这样就保证了不会在同一时刻出现多个线程访问共享资源。

CCriticalSection类的用法如下:
定义CCriticalSection类的一个全局对象(以使各个线程均能访问),如CCriticalSection critical_section;
在访问需要保护的资源或代码之前,调用CCriticalSection类的成员Lock()获得临界区对象: critical_section.Lock();
在线程中调用该函数来使线程获得它所请求的临界区。如果此时没有其它线程占有临界区对象,则调用Lock()的线程获得临界区;否则,线程将被挂起,并放入到一个系统队列中等待,直到当前拥有临界区的线程释放了临界区时为止。
访问临界区完毕后,使用CCriticalSection的成员函数Unlock()来释放临界区: critical_section.Unlock();
再通俗一点讲,就是线程A执行到critical_section.Lock();语句时,如果其它线程(B)正在执行critical_section.Lock();语句后且critical_section. Unlock();语句前的语句时,线程A就会等待,直到线程B执行完critical_section. Unlock();语句,线程A才会继续执行
其实我就是在网上搜了上面这段话,c++也是3天前才开始学的,对它的多线程也不了解。然后花了2小时来折腾,终于写出来了一个能够测试CCriticalSection的代码。
代码测试环境vs2008,win7

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
// Win32ConsoleTest.cpp : Defines the entry point for the console application.
//
 
#include "stdafx.h"
#include
#include //vs2005以上版本都是这么写的
#include
#include
 
CCriticalSection MyCriticalSection;//临界区类用来锁定线程对公用变量的访问
DWORDWINAPI FunctionAdd(LPVOIDlpParameter);//thread data 一定要先声明
DWORDWINAPI FunctionDeduct(LPVOIDlpParameter);//thread data
intnum=100; //计数器
 
int_tmain(intargc, _TCHAR* argv[])
{
HANDLEhThread1;
HANDLEhThread2;
//创建线程
hThread1=CreateThread(NULL,0,FunctionAdd,NULL,0,NULL);
hThread2=CreateThread(NULL,0,FunctionDeduct,NULL,0,NULL);
 
getch();//按任意键退出
return0;
}
 
DWORDWINAPI FunctionAdd(PVOIDarg)//加法函数
{
while(1)
{
MyCriticalSection.Lock();
num++;
printf("执行了num++:%d\n",num);
MyCriticalSection.Unlock();
Sleep(100);//注意这个Sleep一定要放到Lock锁外面,在Lock是不会释放线程的
}
}
 
DWORDWINAPI FunctionDeduct(PVOIDarg)//减法函数
{
while(1)
{
MyCriticalSection.Lock();
num--;
printf("执行了num--:%d\n",num);
MyCriticalSection.Unlock();
Sleep(100);
}
}

说明FunctionAdd,FunctionDeduct都被得到了执行。如果我们把上面代码中的MyCriticalSection.Unlock();注释掉,则会出现全部是num++这种情况。调试跟踪发现不Unlock解锁则一直在FunctionAdd线程里面执行。这也是为什么Sleep()一定要放到Unlock()外面的原因。Lock()会占着线程不放,直到Unlock()

转载请注明http://www.duguxue.com/?p=317



原创粉丝点击