Posix条件变量

int pthread_cond_init(pthread_cond_t *cond, pthread_condattr_t *cond_attr);
int pthread_cond_destroy(pthread_cond_t *cond);int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex);
int pthread_cond_timedwait(pthread_cond_t *cond,  pthread_mutex_t  *mutex,  const  struct timespec *abstime);int pthread_cond_signal(pthread_cond_t *cond);
int pthread_cond_broadcast(pthread_cond_t *cond);

与互斥锁不同,条件变量是用来等待而不是用来上锁的。条件变量用来自动阻塞调用线程, 直到条件变量所要求的情况发生为止。通常条件变量需要和互斥锁同时使用, 利用互斥量保护条件变量;

条件的检测是在互斥锁的保护下进行的。如果一个条件为假,一个线程自动阻塞,并释放等待状态改变的互斥锁。如果另一个线程改变了条件,它就发送信号给关联的条件变量, 并唤醒一个或多个等待在该条件变量上的线程,这些线程将重新获得互斥锁,重新评价条件。如果将条件变量放到共享内存中, 而两进程可共享读写这段内存,则条件变量可以被用来实现两进程间的线程同步。

条件变量使用规范

1.等待条件代码

pthread_mutex_lock(&mutex);while (条件为假)
{pthread_cond_wait(&cond, &mutex);
}
修改条件pthread_mutex_unlock(&mutex);

/**解释: 为什么使用while, 而不用if?

Man-Page给出了答案: If a signal is delivered to a thread waiting for a condition variable, upon return from

the signal handler the thread resumes waiting for the condition variable as if it was not interrupted, or

it shall return zero due to spurious wakeup.

即是说如果正在等待条件变量的一个线程收到一个信号,从信号处理函数返回的时候线程应该重新等待条件变量就好象没有被中断一样,或者被虚假地唤醒返回0。如果是上述情形,那么其实条件并未被改变,那么此时如果没有继续判断一下条件的真假就继续向下执行的话,修改条件将会出现问题,所以需要使用while 循环再判断一下,如果条件还是为假必须继续等待。

注:在多处理器系统中,pthread_cond_signal 可能会唤醒多个等待条件的线程,这也是一种spurious wakeup。

**/

2.给条件发送信号代码

pthread_mutex_lock(&mutex);设置条件为真
pthread_cond_signal(&cond);pthread_mutex_unlock(&mutex);

条件变量API说明

1.pthread_cond_init

使用条件变量之前要先进行初始化:可以在单个语句中生成和初始化一个条件变量如:

pthread_cond_t my_condition=PTHREAD_COND_INITIALIZER; //用于进程间线程的通信;

或用函数pthread_cond_init进行动态初始化;

2.pthread_cond_destroy

该函数可以用来摧毁所指定的条件变量,同时将会释放所给它分配的资源。调用该函数的进程并不要求等待在参数所指定的条件变量上;

3.pthread_cond_wait && pthread_cond_timedwait

cond_wait原语完成三件事:

(1)对mutex解锁;

(2)等待条件, 直到有线程向他发送通知;

(3)当wait返回时, 再对mutex重新加锁;

第一个参数cond是指向一个条件变量的指针。第二个参数mutex则是对相关的互斥锁的指针。

函数pthread_cond_timedwait函数类型与函数pthread_cond_wait区别在于:timedwait多了一个超时, 超时值制订了我们愿意等待多长时间, 如果达到或是超过所引用的参数*abstime,它将结束阻塞并返回错误ETIME.

//timespec结构如下:
struct timespec
{time_t   tv_sec;        /* seconds */long     tv_nsec;       /* nanoseconds */
};

注意: 这个时间值是一个绝对数而不是相对数, 例如, 假设愿意等待三秒钟, 那么并不是把3秒钟转换成timespec结构, 而是需要将当前实践加上3分钟再转换成timespec结构, 这个获取当前时间值的函数可以是clock_gettime(我们采用这一个)也可以是gettimeofday.

4.pthread_cond_signal && pthread_cond_broadcast

cond_signal原语所完成的操作:

向第一个等待条件的线程发起通知, 如果没有任何一个线程处于等待条件的状态, 那么这个通知将被忽略;

cond_broadcast:

向所有等待在该条件上的线程发送通知;

参数cond是一个条件变量的指针。当调用signal时, 一个在相同条件变量上阻塞的线程将被解锁。如果同时有多个线程阻塞,则由调度策略确定接收通知的线程。如果调用broadcast,则将通知阻塞在这个条件变量上的所有线程。一旦被唤醒,线程仍然会要求互斥锁。如果当前没有线程等待通知,则上面两种调用实际上成为一个空操作, 内核会将条件变量的通知忽略(如果参数*cond指向非法地址,则返回值EINVAL);

类Condition封装

//Condition类设计
class Condition
{
public:Condition(const pthread_mutexattr_t *mutexAttr = NULL,const pthread_condattr_t  *condAttr = NULL);~Condition();//条件变量函数int signal();int broadcast();int wait();int timedwait(int seconds);//互斥量函数int lock();int trylock();int unlock();private:pthread_mutex_t m_mutex;pthread_cond_t  m_cond;
};
//Condition类实现
Condition::Condition(const pthread_mutexattr_t *mutexAttr,const pthread_condattr_t  *condAttr)
{//初始化互斥量pthread_mutex_init(&m_mutex, mutexAttr);//初始化条件变量pthread_cond_init(&m_cond, condAttr);
}
Condition::~Condition()
{//销毁互斥量pthread_mutex_destroy(&m_mutex);//销毁条件变量pthread_cond_destroy(&m_cond);
}
int Condition::signal()
{return pthread_cond_signal(&m_cond);
}
int Condition::broadcast()
{return pthread_cond_broadcast(&m_cond);
}
int Condition::wait()
{return pthread_cond_wait(&m_cond, &m_mutex);
}
int Condition::timedwait(int seconds)
{//获取当前时间struct timespec abstime;clock_gettime(CLOCK_REALTIME, &abstime);//将当前时间加上需要等待的秒数, 构成绝对时间值abstime.tv_sec += seconds;return pthread_cond_timedwait(&m_cond, &m_mutex, &abstime);
}int Condition::lock()
{return pthread_mutex_lock(&m_mutex);
}
int Condition::trylock()
{return pthread_mutex_trylock(&m_mutex);
}
int Condition::unlock()
{return pthread_mutex_unlock(&m_mutex);
}

生产者消费者问题(无界缓冲区)

/** 实现: 我们假设是缓冲区是无界的
说明:生产者可以不停地生产,使用pthread_cond_signal  发出通知的时候,如果此时没有消费者线程在等待条件,那么这个通知将被丢弃,但也不影响整体代码的执行,没有消费者线程在等待,说明产品资源充足,即while 判断失败,不会进入等待状态,直接消费产品(即修改条件)。
**/
const unsigned int PRODUCER_COUNT = 5; //生产者个数
const unsigned int CONSUMER_COUNT = 3; //消费者个数//定义Condition类
Condition cond;
//缓冲区 ~O(∩_∩)O~
int nReady = 0;
//消费者
void *consumer(void *args)
{int id = *(int *)args;delete (int *)args;while (true){cond.lock();    //锁定mutexwhile (!(nReady > 0)){printf("-- thread %d wait...\n", id);cond.wait();    //等待条件变量}printf("** thread %d alive, and consume product %d ...\n", id, nReady);-- nReady;  //消费printf("   thread %d end consume... \n\n", id);cond.unlock();  //解锁mutexsleep(1);}pthread_exit(NULL);
}//生产者
void *producer(void *args)
{int id = *(int *)args;delete (int *)args;while (true){cond.lock();    //锁定mutexprintf("++ thread %d signal, and produce product %d ...\n", id, nReady+1);++ nReady;      //生产cond.signal();  //发送条件变量信号printf("   thread %d end produce, signal...\n\n", id);cond.unlock();  //解锁mutexsleep(1);}pthread_exit(NULL);
}int main()
{pthread_t thread[PRODUCER_COUNT+CONSUMER_COUNT];//首先生成消费者for (unsigned int i = 0; i < CONSUMER_COUNT; ++i)pthread_create(&thread[i], NULL, consumer, new int(i));sleep(1);   //使生产者等待一段时间, 加速消费者等待事件产生//然后生成生产者for (unsigned int i = 0; i < PRODUCER_COUNT; ++i)pthread_create(&thread[CONSUMER_COUNT+i], NULL, producer, new int(i));for (unsigned int i = 0; i < PRODUCER_COUNT+CONSUMER_COUNT; ++i)pthread_join(thread[i], NULL);
}

转载于:https://www.cnblogs.com/itrena/p/5926946.html

Linux多线程实践(8) --Posix条件变量解决生产者消费者问题相关推荐

  1. Linux多线程实践(六)使用Posix条件变量解决生产者消费者问题

    前面的一片文章我们已经讲过使用信号量解决生产者消费者问题.那么什么情况下我们须要引入条件变量呢? 这里借用  http://www.cnblogs.com/ngnetboy/p/3521547.htm ...

  2. Linux多线程实践(6) --Posix读写锁解决读者写者问题

    Posix读写锁 int pthread_rwlock_init(pthread_rwlock_t *restrict rwlock,const pthread_rwlockattr_t *restr ...

  3. 操作系统上机作业--使用条件变量解决生产者、计算者、消费者问题(多线程)

    pc1.c: 使用条件变量解决生产者.计算者.消费者问题 /* • 系统中有3个线程:生产者.计算者.消费者 • 系统中有2个容量为4的缓冲区:buffer1.buffer2 • 生产者生产'a'.' ...

  4. Linux多线程实践(5) --Posix信号量与互斥量解决生产者消费者问题

    Posix信号量 Posix 信号量 有名信号量 无名信号量 sem_open sem_init sem_close sem_destroy sem_unlink sem_wait sem_post ...

  5. Linux多线程编程四(条件变量)

    2019独角兽企业重金招聘Python工程师标准>>> 前一节中我们讲述了如何使用互斥锁来实现线程间数据的共享和通信,互斥锁一个明显的缺点是它只有两种状态:锁定和非锁定.而条件变量通 ...

  6. Linux多线程开发-线程同步-条件变量pthread_cond_t

    1.条件变量的概念 一个线程A的执行需要另一个线程B来唤醒,否则A挂起等待.线程B可以产生线程A继续执行的信号.条件变量常用在共享数据状态变化的场景中,例如:生产则和消费者问题.POSIX线程库提供了 ...

  7. 并发编程(一): POSIX 使用互斥量和条件变量实现生产者/消费者问题

    boost的mutex,condition_variable非常好用.但是在Linux上,boost实际上做的是对pthread_mutex_t和pthread_cond_t的一系列的封装.因此通过对 ...

  8. 【操作系统/OS笔记13】信号量、PV操作、管程、条件变量、生产者消费者问题

    本次笔记内容: 10.1 背景 10.2 信号量 10.3 信号量的使用 10.4 信号量的实现 10.5 管程 文章目录 信号量的提出背景 信号量(semaphore) 信号量数据类型 信号量类似铁 ...

  9. 条件变量(生产者消费者问题)

    问题描述 生产者消费者问题:(英语:Producer-consumer problem),也称有限缓冲问题(英语:Bounded-buffer problem),是一个多线程同步问题的经典案例.该问题 ...

最新文章

  1. String 方法中 replace 和 replaceAll 的区别详解(源码分析)
  2. C#编程利器之四:委托与事件(Delegate and event) (上)
  3. Apache用户认证配置文件
  4. 五天带你学完《计算机网络》·第三天·传输层
  5. 悲观锁和乐观锁的详细分析
  6. 以智能数据架构,挖掘增长金矿 1
  7. DEBUG日记:同样的代码在不同的环境却得出不同的结果。
  8. Android http 的使用
  9. linux 下自动重启tomcat的脚本(支持shell杀进程)
  10. windows server 2008中IIS7的功能模塊
  11. 710. Random Pick with Blacklist - LeetCode
  12. sql server数据库中 smallint, int ,bigint ,tinyint的区别与长度
  13. centos 虚拟机glibc升级_CentOS 6 glibc升级到2.15
  14. python在编程语言中排第几_世界公认的十大编程语言,Python居然排第二
  15. FISCO BCOS最强学习路径,汇聚全网资源(2022更新版)
  16. ubuntu18.04系统无法正常连接网络解决办法
  17. VUE3模板,JSX,JSV
  18. GBase 8c 技术白皮书 六
  19. 【解决方案】快递代收点部署视频监控,EasyCVR视频融合平台来助力
  20. 新闻发布系统之分页查询and评论

热门文章

  1. python爬虫案例-Python爬取租房数据实例,据说可以入门爬虫的小案例!
  2. php和python web开发-Web开发应该学习php还是python
  3. python浪漫代码-python七夕浪漫表白源码
  4. 如何自学python基础-零基础如何学习python
  5. python词汇-基本 Python 词汇
  6. python培训班一般多少钱-广州Python培训机构一般多少钱
  7. python3菜鸟-Python3 迭代器与生成器
  8. 用python画简单房子-python绘图作业:使用pygame库画房子
  9. python代码大全p-代码这样写更优雅(Python版)
  10. python语言编程流程图-用流程图带你更好理解python语法