编写同步队列时,有用到条件变量,对操作队列的线程进行同步。当队列为空时,允许get线程挂起,直到add线程向队列添加元素并通过唤醒条件变量,get线程继续向下运行。条件变量在多线程程序中用来实现“等待->唤醒”逻辑常用的方法。条件变量要和互斥量相联结,以避免出现条件竞争:一个线程预备等待一个条件变量,当它在真正进入等待之前,另一个线程恰好触发了该条件。使用条件变量进行同步时,通常以如下方式进行编码:

 pthread_mutex_t mutex;
pthread_cond_t cond; //Get线程
int Get()
{
pthread_mutex_lock(&mutex);
while (queue.empty())
{
pthread_cond_wait(&cond, &mutex);
}
item = queue.front();
pthread_mutex_unlock(&mutex);
return item;
}
//Add线程
void Add(int item)
{
pthread_mutex_lock(&mutex);
queue.push(item);
pthread_cond_signal(&cond);
pthread_mutex_unlock(&mutex);
}

  pthread_cond_wait函数在把线程放进阻塞队列后,自动对mutex进行解锁,使得其它线程可以获得加锁的权利。注意这是一个原子操作,不会造成条件检查和线程进入休眠状态等待之间有其他线程进入的问题,并使当前线程阻塞在cond参数指向的条件变量上,当此线程因条件不满足而进入休眠后,因为互斥量已经解锁,所以别的线程可以对互斥量加锁并改变临界资源,从而唤醒当前阻塞的线程。Add线程获取到锁之后,队列中添加元素,并发送信号唤醒阻塞在条件变量上的Get线程。当有多个Get线程阻塞在条件变量上时,可能会出现徐假唤醒,即阻塞的多个线程都认为自己已经满足条件了,而实际上可能队列中的元素已经被第一个唤醒的线程Get走了,所以这里必须用while(queue.empty())而不是if(queue.empty())进行判断。

  不知道大家有没有注意到在Add函数中line21和line22,pthread_cond_signal和pthread_mutex_unlock的顺序,即唤醒和解锁的先后顺序是否对程序的运行有何影响。这里简单的分析下,我们知道Add线程向队列中添加元素之后,会调用pthread_cond_signal,将阻塞在Get操作的线程唤醒,那么此时阻塞在pthread_cond_wait() 处的线程将苏醒,并进行返回,这里要注意,并非Add线程调用pthread_cond_signal函数后,pthread_cond_wait() 就立即返回,而是在它获取到mutex并重新lock,才能返回继续向下执行。根据分析想到的结论是,如果先signal再unlock,那么当前阻塞在pthread_cond_wait上的线程可能会立即获取到mutex(因为可能有其他Add线程阻塞在mutex上,所以并不一定获取到mutex),重新lock,返回后向下执行;而如果先unlock再signal,那么解锁后mutex可能已经被其他线程获取到了,那么signal唤醒pthread_cond_wait就会继续等待mutex重新被获取走的线程释放,可能会比前一种情况等待的时间长些,由其他的Add线程将其唤醒。不太确定分析的是否正确,于是在Stack Overflow上看到了同样的问题,建议将signal放在unlock之前,部分翻译总结如下:

  如果Add函数将unlock在前,signal在后,可能会产生Spurious wakeups,考虑如下场景:

  1. 线程A阻塞在Get函数,解锁,等待Add操作向队列中添加item。

  2. 线程B调用Add函数,向队列中添加item。在Add函数unlock之后,还未signal之前,发送了上下文切换。

  3. 线程C获取到mutex,调用Add函数,向队列中添加item,解锁并且调用signal函数。

  4. 此时线程A获取到mutex,wait函数返回,处理了刚才Add进来的两个item,之后继续阻塞在条件变量上。

  5. 此时如果线程B得到CPU时间片,那么继续从2处运行,调用signal,唤醒A线程

  6. 线程A被唤醒,但是因为之前的item已经被它Get出来,所以此时队列仍然为空,所以线程A再次进入阻塞状态。

  参考:

  http://stackoverflow.com/questions/6312342/pthread-cond-wait-and-mutex-requirement

  http://stackoverflow.com/questions/1640389/pthreads-pthread-cond-signal-from-within-critical-section

  http://stackoverflow.com/questions/6419117/signal-and-unlock-order

  http://www.domaigne.com/blog/computing/condvars-signal-with-mutex-locked-or-not/#fig01

条件变量signal与unlock的顺序的更多相关文章

  1. 条件变量用例--解锁与signal的顺序问题

    我们知道,当调用signal/broadcast唤醒等待条件变量的其他线程时,既可以在加锁的情况下调用signal/broadcast,也可以在解锁的情况下调用. 那么,到底哪种情况更好呢?man手册 ...

  2. 深入浅出 Java Concurrency (9): 锁机制 part 4 锁释放与条件变量 (Lock.unlock And Condition)

    本小节介绍锁释放Lock.unlock(). Release/TryRelease unlock操作实际上就调用了AQS的release操作,释放持有的锁. public final boolean ...

  3. linux多线程同步pthread_cond_XXX条件变量的理解

    在linux多线程编程中,线程的执行顺序是不可预知的,但是有时候由于某些需求,需要多个线程在启动时按照一定的顺序执行,虽然可以使用一些比较简陋的做法,例如:如果有3个线程 ABC,要求执行顺序是A-- ...

  4. 转: 【Java并发编程】之二十:并发新特性—Lock锁和条件变量(含代码)

    简单使用Lock锁 Java5中引入了新的锁机制--Java.util.concurrent.locks中的显式的互斥锁:Lock接口,它提供了比synchronized更加广泛的锁定操作.Lock接 ...

  5. 深入解析条件变量(condition variables)

    深入解析条件变量 什么是条件变量(condition variables) 引用APUE中的一句话: Condition variables are another synchronization m ...

  6. 学习pthreads,使用条件变量进行多线程之间的同步

    条件变量提供另一种多线程同步的方法.互斥量通过控制对共享数据的访问来同步任务.条件变量可以根据数据的值来同步任务.条件变量是当一个事件发生时发送信号的信号量.一旦事件发生,可能会有多个线程在等待信号, ...

  7. 线程同步,条件变量pthread_cond_wait

    与互斥锁不同,条件变量是用来等待而不是用来上锁的.条件变量用来自动阻塞一个线程,直到某特殊情况发生为止.条件变量使我们可以睡眠等待某种条件出现.条件变量是利用线程间共享的全局变量进行同步的一种机制,主 ...

  8. 【Java并发编程】:并发新特性—Lock锁和条件变量

    简单使用Lock锁 Java5中引入了新的锁机制——Java.util.concurrent.locks中的显式的互斥锁:Lock接口,它提供了比synchronized更加广泛的锁定操作.Lock接 ...

  9. 多线程编程中条件变量和的spurious wakeup 虚假唤醒

    1. 概述 条件变量(condition variable)是利用共享的变量进行线程之间同步的一种机制.典型的场景包括生产者-消费者模型,线程池实现等. 对条件变量的使用包括两个动作: 1) 线程等待 ...

随机推荐

  1. SQLite外键

    数据库工具:SQLite Manager(V0.7.7) SQLite版本号:V3.6.19+ SQLite Manager 默认是不开启外键的. 那么怎样,使用它创建一个带有外键的表呢? 一.开启外 ...

  2. 它们的定义UIAlertView

    code4App有很多伟大的上方UI特效代码,,好牛逼啊,这效果,太炫了,哇,怎么自己写不出来.事实上,再炫的特效,都是依据苹果系统的框架而来,假设我们了解系统框架实现的原理,也就能写出属于自己自己定 ...

  3. 创建Windows类别

    Windows在表单.控制.对话框基本上形成.Windows类是Windows形式的类型,可处理叙述性说明. 在Windows提前有很多定义Windows类别,但它可以很容易地创建自己的Windows ...

  4. twitter接口开发

    前一阵子研究了下twitter接口,发现网上的资料不是很多.遂花了些心血,终于有所收获~ 现在有时间赶紧整理出来便于自己以后查阅,也想帮助有困难的同学们.废话不多说,现在就以最简洁的方式开始了.注意: ...

  5. java 7K交通灯管理系统面试题

    交通灯管理系统 模拟实现十字路口的交通灯管理系统逻辑.详细需求例如以下: 1. 异常随机生成依照各个路线行驶的车辆.    比如:    由南向而来去往北向的车辆----直行车辆    由西向而来去往 ...

  6. ubunut 查看port被哪个程序占用

    查看8087port被哪个程序占用 lsof -i :8087 -n

  7. 大数据系列修炼-Scala课程05

    Scala多重继承.构造器的执行顺序.AOP实现 多重继承的trait实现:Scala中接口可以继承具体的类,trait接口可以实现多重继承,并且某个类也可以继承特定的类,在继承后面可以混入,接口的实 ...

  8. 项目中经常使用的JS方法汇总,非常有用

    // 对Date的扩展,将 Date 转化为指定格式的String   // 月(M).日(d).小时(h).分(m).秒(s).季度(q) 可以用 1-2 个占位符,   // 年(y)可以用 1- ...

  9. ROOT android 原理。 基于(zergRush)

    出自: http://bbs.gfan.com/android-2996211-1-1.html 须要ROOT的同学请去上面的地址下载. a.控制手机创建个暂时目录,然后把zergRush脚本写入此目 ...

  10. 【百度地图API】建立全国银行位置查询系统(五)——如何更改百度地图的信息窗口内容?

    原文:[百度地图API]建立全国银行位置查询系统(五)--如何更改百度地图的信息窗口内容? 摘要: 酷讯.搜房.去哪儿网等大型房产.旅游酒店网站,用的是百度的数据库,却显示了自定义的信息窗口内容,这是 ...