Windows用户模式下的线程同步之CRITICAL_SECTION(关键字)

关键段的的方式使多行代码以"原子方式"来对资源进行访问。

CRITICAL_SECTION g_cs;//定义关键

EnterCriticalSection(&g_cs);
//处理多个线程共同访问的资源
LeaveCriticalSection(&g_cs);

在使用EnterCriticalSection之前必须调用InitialiseCriticalSection函数初始化g_cs,当知道线程不再需要访问共享资源的时候的应该调用DeleteCriticalSection函数清理g_cs。

使用未初始化的g_cs会发生不可预知的错误。

关键字最大的好处是容易使用,在内部使用了InterLocked(原子锁)函数,执行速度非常快,最大缺点是无法用来在多个进程之间对多个线程同步。


如果没有现成正在访问资源,EnterCriticalSection会更新g_cs结构体中的成员变量,并立即返回,线程访问共享资源。

如果有一个线程正在访问共享资源,那么EnterCriticalSection会使用一个内核对象把调用线程切换到等待状态。一旦当前正在访问共享资源的线程调用LeaveCriticalSection

系统会自动更新g_cs的成员变量,并将等待状态的线程切换回可调度状态。

在多核CPU的情况下如果两个线程同时调用EnterCriticalSection,一个线程将获得访问的权限,另一个切换到等待状态。

将线程切换成等待状态实际是将线程从用户模式切换成内核模式,大约1000个cpu周期。事实上在在需要等待的线程完全切换到内核模式之前,访问资源的线程可能已经调用LeaveCriticalSection函数了。出现这种情况会大量浪费CPU时间。在这种情况下可以同时使用旋转锁。

BOOL InitialiseCriticalSectionAndSpinCount()函数,第二个参数是旋转次数。还可以调用SetCriticalSectionSpinCount()来设置旋转次数


Slim读/写锁

SRWLock,与关键段不同的是,SRWLock允许我们区分读取者线程和写入者线程。可以让所有读取者线程在同一时刻访问共享资源,只有当写入者线程需要更新共享的资源的时候才需要进行同步,在这种情况下写入者线程会独占共享资源,其它线程无权访问。

使用SRWLock,首先分配一个SRWLock结构 srwlock,调用InitialiseSRWLock(&srwlock),进行初始化。

在写入者线程中调用AcquireSRWLockExclusive(&srwlock),获得共享资源的独占权,写入完成后需要调用ReleaseSRWLockExclusive(&srwlock)解除独占权。

在读取线程中调用ACquireSRWLockShared(&srwlock);和ReleaseSrwLockShared(&srwlock)。

相比关键段 ,如果锁已经被占用,那么再调用AcquireSRWLockExclusive(shared)会使线程阻塞,而且一个线程不能为了多次写入资源而多次调用AcquireSRWLockExclusive后在多次调用ReleaseSRWLockExclusive来释放。


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值