为什么线程同步的时候pthread_cond_t要和pthread_mutex_t同时使用

转自:http://blog.sina.com.cn/s/blog_6ffd3b5c0100mc3n.html

pthread_mutex_t count_lock;
pthread_cond_t count_nonzero;
unsigned count = 0;

decrement_count () {
       pthread_mutex_lock (&count_lock);
       while(count==0)
               pthread_cond_wait( &count_nonzero, &count_lock);//此函数会先解锁,再阻塞等待条件变量,最后加锁

       count=count -1;
       pthread_mutex_unlock (&count_lock);
}
 
increment_count(){
       pthread_mutex_lock(&count_lock);
 
       if(count==0)
               pthread_cond_signal(&count_nonzero);
 
       count=count+1;
       pthread_mutex_unlock(&count_lock);
}

decrement_count和increment_count在两个线程A和B中被调用。
正确的情况下,如果decrement_count首先运行,那么A会被阻塞到pthread_cond_wait。随后increment_count运行,它调用pthread_cond_signal唤醒等待条件锁count_nonzero的A线程,但是A线程并不会马上执行,因为它得不到互斥锁count_lock。当B线程执行pthread_mutex_unlock之后A线程才得以继续执行。
 
如果pthread_cond_signal前后没有使用互斥锁count_lock保护,可能的情况是这样。A阻塞到pthread_cond_wait,然后B执行到pthread_cond_signal时候,发生了线程切换,于是A被唤醒,并且发现count依然是0,所以继续阻塞到条件锁count_nonzero上。然后B继续执行,这时候尽管count=1,A永远不会被唤醒了。这样就发生了逻辑错误。
 
当然在这个上下文中,如果把count=count+1放在函数放在pthread_cond_signal之前变成
 
increment_count(){
         count=count+1;
 
       if(count==0)
               pthread_cond_signal(&count_nonzero);
}
 
这样没有问题。但是这种方法并不能保证所有情况下都适用。于是需要用互斥锁保护条件锁相关的变量。也就是说条件锁是用来线程通讯的,但是互斥锁是为了保护这种通讯不会产生逻辑错误,可以正常工作。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值