互斥量和条件变量配合的典型应用

声明:该文从网上找到的,写的不错

 

关于pthread_cond_signal函数

pthread_cond_signal函数的作用是发送一个信号给另外一个正在处于阻塞等待状态的线程,使其脱离阻塞状态,继续执行.如果没有线程处在阻塞等待状态,pthread_cond_signal也

 

会成功返回。

但使用pthread_cond_signal不会有“惊群现象”产生,他最多只给一个线程发信号。假如有多个线程正在阻塞等待着这个条件变量的话,那么是根据各等待线程优先级的高低确定

 

哪个线程接收到信号开始继续执行。如果各线程优先级相同,则根据等待时间的长短来确定哪个线程获得信号。但无论如何一个pthread_cond_signal调用最多发信一次。

另外,互斥量的作用一般是用于对某个资源进行互斥性的存取,很多时候是用来保证操作是一个原子性的操作,是不可中断的。

pthread_cond_signal函数与条件变量的典型应用就是用来实现producer/consumer模型。

 

 

[code]

#include <stdio.h>

#include <stdlib.h>

#include <unistd.h>

#include <pthread.h>

 

#define BUFFER_SIZE 8

 

struct Products

{

int buffer[BUFFER_SIZE];

pthread_mutex_t locker;          //保证存取操作的互斥性

pthread_cond_t notEmpty; //是否可读       

pthread_cond_t notFull;          //是否可写

int posReadFrom;

int posWriteTo;

};

 

int BufferIsFull(struct Products* products)

{

if ((products->posWriteTo + 1) % BUFFER_SIZE == products->posReadFrom)

{

return (1);

}

return (0);

}

int BufferIsEmpty(struct Products* products)

{

if (products->posWriteTo == products->posReadFrom)

{

return (1);

}

return (0);

}

 

//制造产品

void Produce(struct Products* products, int item)

{

int empty;

 

//原子操作! 错!! 应该是得不到锁,进入阻塞状态,等待其他线程释放该锁。

pthread_mutex_lock(&products->locker); 

 

//释放锁,这样消费线程才可以访问资源,并消费,之后通知notfull消息

while (BufferIsFull(products)){

pthread_cond_wait(&products->notFull, &products->locker); 

} //无空间可写入

 

         //判断是否为空,设置empty标志

empty = BufferIsEmpty( products);

 

//写入数据

products->buffer[products->posWriteTo] = item;

products->posWriteTo++;

if (products->posWriteTo >= BUFFER_SIZE)

products->posWriteTo = 0;

 

//如果之前为空,则发通知给消费线程

if(empty){

pthread_cond_signal(&products->notEmpty);   

}

pthread_mutex_unlock(&products->locker);    //解锁

}

 

//消费产品

int Consume(struct Products* products)

{

int item, full;

pthread_mutex_lock(&products->locker);

 

// 为空时持续等待,无数据可读,

// 进入等待状态,不占用CPU,同时,释放占用的互斥锁,一旦被唤醒则再次锁定互斥量,即锁定product结构体的读写。

while (BufferIsEmpty(products))

{

pthread_cond_wait(&products->notEmpty, &products->locker);

 

//判断是否为满,设置full标志

full = BufferIsFull(products)

 

//提取数据

item = products->buffer[products->posReadFrom];

products->posReadFrom++;

if (products->posReadFrom >= BUFFER_SIZE) //如果到末尾,从头读取

products->posReadFrom = 0;

 

         //如果之前是full的,则发信号通知生产线程

         if(full){

pthread_cond_signal(&products->notFull); 

}

pthread_mutex_unlock(&products->locker);

 

return item;

}

 

#define END_FLAG (-1)

struct Products products;

 

void* ProducerThread(void* data)

{

int i;

for (i = 0; i < 16; ++i)

{

printf("producer: %d/n", i);

Produce(&products, i);

}

Produce(&products, END_FLAG);

return NULL;

}

void* ConsumerThread(void* data)

{

int item;

while (1)

{

item = Consume(&products);

if (END_FLAG == item)

       break;

printf("consumer: %d/n", item);

}

return (NULL);

}

 

int main(int argc, char* argv[])

{

pthread_t producer;

pthread_t consumer;

int result;

 

products.notEmpty = PTHREAD_COND_INITIALIZER;

products.notFull  = PTHREAD_COND_INITIALIZER;

products.locker   = PTHREAD_MUTEX_INITIALIZER;

 

pthread_create(&producer, NULL, &ProducerThread, NULL);

pthread_create(&consumer, NULL, &ConsumerThread, NULL);

pthread_join(producer, (void *)&result);

pthread_join(consumer, (void *)&result);

 

return 0;

}

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值