线程调用 | 描述 |
pthread_cond_init | 创建一个条件变量 |
pthread_cond_destroy | 撤销一个条件变量 |
pthread_cond_wait | 阻塞以等待一个信号 |
pthread_cond_signal | 向另一个线程发送信号来唤醒它 |
pthread_cond_broadcast | 向多个线程发信号来让他们全部唤醒 |
pthread_cond_wait函数
pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex)函数传入的参数mutex用于保护条件,因为我们在调用pthread_cond_wait时,如果条件不成立我们就进入阻塞,但是进入阻塞这个期间,如果条件变量改变了的话,那我们就漏掉了这个条件。因为这个线程还没有放到等待队列上,所以调用pthread_cond_wait前要先锁互斥量,即调用pthread_mutex_lock(),pthread_cond_wait在把线程放进阻塞队列后,自动对mutex进行解锁,使得其它线程可以获得加锁的权利。这样其它线程才能对临界资源进行访问并在适当的时候唤醒这个阻塞的进程。当pthread_cond_wait返回的时候又自动给mutex加锁。
转于:http://sgy618.itpub.net/post/42829/513424
pthread_cond_signal函数:
pthread_cond_signal函数的作用是发送一个信号给另外一个正在处于阻塞等待状态的线程,使其脱离阻塞状态,继续执行.如果没有线程处在阻塞等待状态,pthread_cond_signal也会成功返回。
但使用pthread_cond_signal不会有“惊群现象”产生,他最多只给一个线程发信号。假如有多个线程正在阻塞等待着这个条件变量的话,那么 是根据各等待线程优先级的高低确定哪个线程接收到信号开始继续执行。如果各线程优先级相同,则根据等待时间的长短来确定哪个线程获得信号。但无论如何一个 pthread_cond_signal调用最多发信一次。另外,互斥量的作用一般是用于对某个资源进行互斥性的存取,很多时候是用来保证操作是一个原子性的操作,是不可中断的。
pthread_cond_signal函数与条件变量的典型应用就是用来实现producer/consumer模型。
生产者-消费者例子(现代操作系统 第三版):
/*************************************************************************
> File Name: mutex_cond.c
> Created Time: Thu 17 Jul 2014 07:27:48 PM CST
************************************************************************/
#include<stdio.h>
#include<pthread.h>
#define MAX 100000000
pthread_mutex_t the_mutex;
pthread_cond_t condc, condp;
int buffer = 0;
void *producer(void *ptr)
{
int i;
for(i = 1; i<=MAX; ++i)
{
pthread_mutex_lock(&the_mutex);
if(buffer != 0) pthread_cond_wait(&condp, &the_mutex);
buffer = i;
printf("%d\n",buffer);
pthread_cond_signal(&condc);
pthread_mutex_unlock(&the_mutex);
}
pthread_exit(0);
}
void *consumer(void *ptr)
{
int i;
for(i = 1; i<=MAX; ++i)
{
pthread_mutex_lock(&the_mutex);
if(buffer == 0) pthread_cond_wait(&condc, &the_mutex);
buffer = 0;
printf("%d\n",buffer);
pthread_cond_signal(&condp);
pthread_mutex_unlock(&the_mutex);
}
pthread_exit(0);
}
int main()
{
pthread_t pro,con;
pthread_mutex_init(&the_mutex, 0);
pthread_cond_init(&condc,0);
pthread_cond_init(&condp,0);
pthread_create(&pro, 0, producer, 0);
pthread_create(&con, 0, consumer, 0);
pthread_join(pro, 0);
pthread_join(con, 0);
pthread_cond_destroy(&condc);
pthread_cond_destroy(&condp);
pthread_mutex_destroy(&the_mutex);
}