C++关于临界区CCriticalSection的线程同步 网狐框架示例

今天早上在gitbook上翻译了msdn文档里的Critical Section Objects,渣渣翻译。

1. 临界区

1.1临界区是什么

临界区与互斥量、事件、信号一样,都提供了一种同步机制。临界区在同一时间内仅能被一个线程所拥有,这对于保护共享资源不被并发访问非常有用。但是它只能在单进程中使用,无法跨进程共享。

1.1.1什么是同步,什么是异步?

同步和异步关注的是消息通信机制
以下的理解来自于知乎(愚抄 严肃 陈硕
下面用两个例子来解释同步和异步:

  • 当你用水壶烧一壶水,此时有两种水壶,一种是烧开不会响笛的水壶A,另一种是烧开会响笛的水壶B。
    当你使用水壶A烧水的时候,你需要自己来看看水有没有烧开,这就是同步
    当你使用水壶B烧水的时候,水烧开了它自动就会通知你,这就是异步

  • 你打电话给书店老板,问有没有《C++ primer》这本书。
    同步机制里,老板会说“稍等,我找找”,然后就开始查找,等查好了就告诉你结果(返回结果)。
    异步机制里,老板会说“我去找一下,查到了再打电话给你”,然后直接挂掉电话(不返回结果),查好了他就主动打电话给你(相当于回调函数)。

需要注意的一个重点:在处理IO的时候,阻塞与非阻塞都是同步IO

同步与异步

1.2 如何使用临界区

当该临界区对象被某个线程占用时,另一个线程想要访问该对象,线程就会进入休眠状态,直到临界区对象被释放,才会唤醒该线程。

因为唤醒线程需要时间,所以现在为了避免性能降低。在另一个线程访问对象且该对象已被占用的时候,设置一个循环访问次数,在这个次数内不断循环访问临界区对象,如果该对象被释放,这个线程就不会进入休眠。如果该对象在循环次数内依旧没有释放,线程就会进入线程。

2. CCriticalSection(临界区)同步对象

2.1 关于CCriticalSection

CCriticalSection是MFC提供的一种同步对象。它是一个用于同步的对象,同一时刻只允许一个线程存取资源或代码区。

2.2 CCriticalSection的用法
  1. 定义一个CCriticalSection类的全局对象,因为是全局对象,那么各个线程均可以访问。
  2. 在访问需要保护的资源或代码之前,调用CCriticalSection类的成员Lock()获得临界区对象
    在线程中调用该函数来使线程获得它所请求的临界区,如果此时没有其他线程占用临界区对象,则调用Lock(),线程获取临界区。否则,线程被挂起,并放入到一个系统队列中等待,直到当前拥有的线程释放了临界区为止。
  3. 访问临界区结束后,使用Unlock()来释放临界区。
2.3 CCriticalSection在网狐框架中的使用
  1. 它为CCriticalSection专门定义了一个新类CWHDataLocker,其中增加了锁定计数这个字段,在使用临界区的时候,不单单是锁定临界区,还会将锁定计数+1,在解锁的时候,会将锁定计数-1
//变量定义
private:
    INT                             m_nLockCount;                   //锁定计数
    CCriticalSection &              m_CriticalSection;              //锁定对象


//构造函数
CWHDataLocker::CWHDataLocker(CCriticalSection & CriticalSection, bool bLockAtOnce) 
    : m_CriticalSection(CriticalSection)
{
    //设置变量
    m_nLockCount=0;

    //锁定对象
    if (bLockAtOnce==true)
    {
        Lock();
    }

    return;
}

//析构函数
CWHDataLocker::~CWHDataLocker()
{
    //解除锁定
    while (m_nLockCount>0)
    {
        UnLock();
    }

    return;
}

//锁定函数
VOID CWHDataLocker::Lock()
{
    //锁定对象
    m_CriticalSection.Lock();

    //设置变量
    m_nLockCount++;

    return;
}

//解锁函数
VOID CWHDataLocker::UnLock()
{
    //效验状态
    ASSERT(m_nLockCount>0);
    if (m_nLockCount==0) return;

    //设置变量
    m_nLockCount--;

    //解除锁定
    m_CriticalSection.Unlock();

    return;
}
  1. 在队列服务中,我们可以看到该类的使用,在队列中加入数据或者提取数据都使用了临界区。因为临界区在函数中实例化,所以在函数结束后会自动调用CWHDataLocker的析构函数解除临界区的锁定
//加入数据
bool  CQueueService::AddToQueue(WORD wIdentifier,void *const pBuffer, WORD wDataSize)
{
    CWHDataLocker lock(m_CriticalSection);//
    m_DataQueue.InsertData(wIdentifier, pBuffer, wDataSize);
    PostQueuedCompletionStatus(m_hCompletionPort, wDataSize, (ULONG_PTR)this, NULL);
    return true;
}

//提取数据
bool CQueueService::GetData(tagDataHead & DataHead, void * pBuffer, WORD wBufferSize)
{
    CWHDataLocker lock(m_CriticalSection);//
    return m_DataQueue.DistillData(DataHead, pBuffer, wBufferSize);
}
  1. 网狐框架中大量使用了临界区:登录服务器、游戏服务器、定时器引擎、队列服务TCP引擎、异步引擎等等
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,376评论 6 491
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,126评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,966评论 0 347
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,432评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,519评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,792评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,933评论 3 406
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,701评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,143评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,488评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,626评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,292评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,896评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,742评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,977评论 1 265
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,324评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,494评论 2 348

推荐阅读更多精彩内容