实验名称:经典同步问题:生成者与消费者问题

文章描述了一个使用信号量和多线程解决的经典同步问题——生产者/消费者问题。通过创建多个生产者和消费者线程,利用信号量(room_sem和product_sem)控制缓冲池的读写,确保生产者在缓冲池未满时可以添加消息,消费者在缓冲池非空时可以取出消息。线程通过pthread_create创建,pthread_join等待线程结束,同时使用pthread_mutex_t进行线程互斥。
摘要由CSDN通过智能技术生成

实验名称:经典同步问题:生成者与消费者问题

相关知识

信号量

信号量是用来协调不同进程间的数据对象,可用来保护共享资源,也能用来实现进程间及同一进程不同线程间的进程同步。分为二值信号灯和计算信号灯两种类型。

进程与线程原语的比较

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tIyhHBPe-1671105369897)(Images/实验:进程管理-进程同步问题1.png)]

线程创建

线程创建是通过函数 pthread_create(thread,attr,start_routine,arg)函数来实现的,而该函数是通过Linux特有的系统调用clone来实现的。
格式:

#include<pthread.h>
int pthread_create(thread,attr,start_routine,arg);

其中参数thread为线程标识符,attr为线程属性设置,start_routine为线程函数起始地址,arg为传递给start_routine的参数。创建线程成功返回0,否则返回错误号。

获得线程标识符

格式:

#include<pthread.h>
pthread_t pthread_self(void);

说明:返回调用的线程的标识符。每个线程都有自己的线程标识符,以便在进程内区分,线程标识符在pthread_create创建时产生。

线程等待

格式:

#include<pthread.h>
int pthread_join(thread,retval);

说明:该函数将调用它的线程阻塞,一直等到被等待的线程结束为止,当函数返回时,被等待线程的资源被收回。thread为被等待的线程标识符,retval为用户定义的指针,存放被等待线程的返回值。

线程退出

格式:

#include<pthread.h>
void pthread_exit(retval); //终止调用线程,`retval`为线程的返回值。

int pthread_cancel(thread); //终止由参数thread指定的线程

实验内容

使用多线程和信号量解决生产者/消费者问题:有一个长度为N的缓冲池被生产者和消费者共同使用。只要缓冲池未满,生产者就可以将消息送入缓冲池;只要缓冲池不空,消费者便可从缓冲池中取走一个消息。生产者向缓冲池放入消息的同时,消费者不能操作缓冲池,反之亦然。

pthread_join()将调用它的线程阻塞,一直等到被等待的线程结束为止,当函数返回时,被等待线程的资源被收回。本实验中使用room_sem信号量来表示缓冲区可用空间,product_sem信号量表示缓冲区中有无可用产品,而mutex代表线程互斥信号量。

编写producer_consumer.c:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<pthread.h>
#include<semaphore.h>
#include<sys/types.h>
#define PRODUCER_NUM 5
#define CONSUMER_NUM 5
#define POOL_SIZE 11
int pool[POOL_SIZE]; //buffer
int head=0; //read pointer
int rear=0; //write pointer
sem_t room_sem;//available room in buffer
sem_t product_sem;//available products in buffer
pthread_mutex_t mutex;
void producer_fun(void*arg)
{
        while(1)
        {
                sleep(1);
                sem_wait(&room_sem);
                pthread_mutex_lock(&mutex);
                //producer write data to buffer
                pool[rear]=1;
                rear=(rear+1)%POOL_SIZE;
                printf("producer %d write to pool\n",(int)arg);
                printf("pool size is %d\n",(rear-head+POOL_SIZE)%POOL_SIZE);
                pthread_mutex_unlock(&mutex);
                sem_post(&product_sem);
        }
}
void consumer_fun(void *arg)
{
        while(1)
        {
                int data;
                sleep(10);
                sem_wait(&product_sem);
                pthread_mutex_lock(&mutex);
                //consumer read data in buffer
                data=pool[head];
                head=(head+1)%POOL_SIZE;
                printf("consumer %d read from pool\n",(int)arg);
                printf("pool size is %d\n",(rear-head+POOL_SIZE)%POOL_SIZE);
                pthread_mutex_unlock(&mutex);
                sem_post(&room_sem);
        }
}
int main()
{
        pthread_t producer_id[PRODUCER_NUM];
        pthread_t consumer_id[CONSUMER_NUM];
        pthread_mutex_init(&mutex,NULL);
        int ret=sem_init(&room_sem,0,POOL_SIZE-1);//initialize the signal room_sem
        if(ret!=0)
        {
                printf("sem_init error\n");
                exit(0);
        }
        ret=sem_init(&product_sem,0,0); //initialize the signal produc_sem
        if(ret!=0)
        {
                printf("sem_init error\n");
                exit(0);
        }
        for(int i=0;i<PRODUCER_NUM;i++)
        {
                //create producer thread
                ret=pthread_create(&producer_id[i],NULL,producer_fun,(void*)i);
                if(ret!=0)
                {
                        printf("producer_id error\n");
                        exit(0);
                }
                //create consumer thread
                ret=pthread_create(&consumer_id[i],NULL,consumer_fun,(void*)i);
                if(ret!=0)
                {
                        printf("consumer_id error\n");
                        exit(0);
                }
        }
        for(int i=0;i<PRODUCER_NUM;i++)
        {
                pthread_join(producer_id[i],NULL);
                pthread_join(consumer_id[i],NULL);
        }
        exit(0);
}

编译时使用以下命令:
gcc -o producer_consumer producer_consumer.c -lpthread

注:编译选项要加上-lpthread,因为pthread不是Linux默认库,链接时需要使用静态库libpthread.a

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

zhugenmi

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值