Linux操作系统实验初学(1)(生产者消费者问题)

数据类型:

sem_t :信号量的数据类型,本质上是一个长整型

   pthread_t:用于声明线程的ID。

pthread_mutex_t:互斥锁

函数:

int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t * attr):用于初始化互斥锁。

int sem_init(sem_t *sem, int pshared, unsigned int value):初始化有sem指向的信号量对象,设置它的共享选项,

并置初始化整数值。参数pshared控制信号量的类型,如果其值为0,就表示该信号量是当前进程的局部信号量,

否则,这个信号量可以在多个进程中共享。

int sem_wait(sem_t *sem);以原子操作的方式将信号量的值减1.

int sem_post(sem_t *sem):以原子操作的方式将信号量的值加1.

int sem_destroy(sem_t *sem):用于清理信号量。

int pthread_mutex_lock(pthread_mutex_t *mutex)

int pthread_mutex_umlock(pthread_mutex_t *mutex)

int pthread_mutex_destroy(pthread_mutex_t *mutex)

int pthread_create(pthread_t *restrict tidp, const pthread_attr_t *restrict attr, void*(*start_rtn)(void), void *restrict arg);

函数返回0表示成功,非0表示失败。

int pthread_join(pthread_t thread, void **status);参数thread为指定将要等待的线程,参数status指向另一个指针的指

针,用于指定返回值。

生产者消费者问题


#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<pthread.h>
#include<semaphore.h>
#define n 11 //缓冲池大小
#define PRODUCER_SIZE 5 //生产者数量
#define CONSUMER_SIZE 5 //消费者数量
int pool[n];//缓冲区
int in = 0; // 缓冲池写入指针
int out = 0;// 缓冲池读出指针
pthread_mutex_t mutex; // 互斥信号量
sem_t empty; //空缓冲池数量
sem_t full; //满缓冲池数量
int id1 = 0, id2 = 0; // 生产者id 消费者id
void *producer(void *arg){
    int id = ++id1;
    while(1){
        sleep(1);
        sem_wait(&empty);
        pthread_mutex_lock(&mutex);
        pool[in] = 1;
        in = (in + 1) % n;
        printf("producer %d write to pool\n", id);
        printf("pool size is %d\n", (in - out + n) % n);
        pthread_mutex_unlock(&mutex);
        sem_post(&full);
    }
}
void *consumer(void *arg){
    int id = ++ id2;
    while(1){
        sleep(10);
        sem_wait(&full);
        pthread_mutex_lock(&mutex);
        out = (out + 1) % n;
        printf("consumer %d read from pool\n", id);
        printf("pool size is %d\n", (in - out + n) % n);
        pthread_mutex_unlock(&mutex);
        sem_post(&empty);
    }
}
int main(){
    pthread_t producer_id[PRODUCER_SIZE];
    pthread_t consumer_id[CONSUMER_SIZE];
    pthread_mutex_init(&mutex, NULL); // 初始化互斥量
    int flag = sem_init(&empty, 0, n - 1); // 初始化信号量empty为缓冲池大小
    if(flag != 0){
        printf("sem_init error\n");
        exit(0);
    }
    flag = sem_init(&full, 0, 0); // 初始化信号量full为0
    if(flag != 0){
        printf("sem_init error\n");
        exit(0);
    }
    for(int i = 0; i < PRODUCER_SIZE; ++i){
        //创建生产者线程
        flag = pthread_create(&producer_id[i], NULL, producer, (void*) (&i));
        if(flag != 0){
            printf("producer_id error\n");
            exit(0);
        }
    }
    for(int i = 0; i < CONSUMER_SIZE; ++i){
        //创建消费者线程
        flag = pthread_create(&consumer_id[i], NULL, consumer, (void*) (&i));
        if(flag != 0){
            printf("consumer_id error\n");
            exit(0);
        }
    }
    for(int i = 0; i < PRODUCER_SIZE; ++i){
	pthread_join(producer_id[i], NULL);
	pthread_join(consumer_id[i], NULL);
    }
	return 0;
}


(1)创建生产者和消费者线程 在Windows2000环境下,创建一个控制台进程,在此进程中创建n个线程来模拟生产者或者消费者。这些线程的信息由本程序定义的“测试用例文件”中予以指定。 该文件的格式和含义如下: 3 1 P 3 2 P 4 3 C 4 1 4 P 2 5 C 3 1 2 4 第一行说明程序中设置几个临界区,其余每行分别描述了一个生产者或者消费者线程的信息。每一行的各字段间用Tab键隔开。不管是消费者还是生产者,都有一个对应的线程号,即每一行开始字段那个整数。第二个字段用字母P或者C区分是生产者还是消费者。第三个字段表示在进入相应线程后,在进行生产和消费动作前的休眠时间,以秒计时;这样做的目的是可以通过调整这一列参数,控制开始进行生产和消费动作的时间。如果是代表生产者,则该行只有三个字段。如果代表消费者,则该行后边还有若干字段,代表要求消费的产品所对应的生产者的线程号。所以务必确认这些对应的线程号存在并且该线程代表一个生产者。 (2)生产和消费的规则 在按照上述要求创建线程进行相应的读写操作时,还需要符合以下要求: ①共享缓冲区存在空闲空间时,生产者即可使用共享缓冲区。 ②从上边的测试数据文件例子可以看出,某一生产者生产一个产品后,可能不止一个消费者,或者一个消费者多次地请求消费该产品。此时,只有当所有的消费需求都被满足以后,该产品所在的共享缓冲区才可以被释放,并作为空闲空间允许新的生产者使用。 ③每个消费者线程的各个消费需求之间存在先后顺序。例如上述测试用例文件包含一行信息“5 C 3 l 2 4”,可知这代表一个消费者线程,该线程请求消费1,2,4号生产者线程生产的产品。而这种消费是有严格顺序的,消费1号线程产品的请求得到满足后才能继续往下请求2号生产者线程的产品。 ④要求在每个线程发出读写操作申请、开始读写操作和结束读写操作时分别显示提示信息。 (3)相关基础知识 本实验所使用的生产者和消费者模型具有如下特点: 本实验的多个缓冲区不是环形循环的,也不要求按顺序访问。生产者可以把产品放到目前某一个空缓冲区中。 消费者只消费指定生产者的产品。 在测试用例文件中指定了所有的生产和消费的需求,只有当共享缓冲区的数据满足了所有关于它的消费需求后,此共享缓冲区才可以作为空闲空间允许新的生产者使用。 本实验在为生产者分配缓冲区时各生产者间必须互斥,此后各个生产者的具体生产活动可以并发。而消费者之间只有在对同一产品进行消费时才需要互斥,同时它们在消费过程结束时需要判断该消费对象是否已经消费完毕并清除该产品。 Windows用来实现同步和互斥的实体。在Windows中,常见的同步对象有:信号量(Semaphore)、互斥量(Mutex)、临界段(CriticalSection)等。使用这些对象都分为三个步骤,一是创建或者初始化:接着请求该同步对象,随即进入临界区,这一步对应于互斥量的上锁;最后释放该同步对象,这对应于互斥量的解锁。这些同步对象在一个线程中创建,在其他线程中都可以使用,从而实现同步互斥。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值