实验:线程同步与互斥

任务一 创造一个多线程共同访问变量的问题
编写程序,定义一个全局变量sum,并赋值为0,在主线程中创建20个子线程,要求
1、这些子线程完成同样的功能,每个线程循环10000次,每次对sum执行+1操作
2、所有子线程执行完之后,在主线程中打印最终的sum的值
提示
每个线程对sum+1执行10000次,总共创建20个线程,那么这些子线程全部执行完之后,sum的值理论上应该是200000,但是,程序运行的结果并非200000(或许你的程序运行结果确实是200000,那么你可以尝试将子线程数目增加,但要注意子线程数目过多可能会导致部分线程创建失败,适当调整你的子线程数目,到能够出现竞争问题为止,即出现sum最终值不符合预期为止)。
hello.c

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<pthread.h>

int sum = 0;

void *thrd_func(void *arg) {
    int i;
    for(i=0; i<10000; i++) {
        sum++;
    }
}

int main(){
    int ret,i;
    pthread_t tid[20];
    for(i=0; i<20; i++){
    ret = pthread_create(&tid[i], NULL,thrd_func, NULL);
    }
    for(i=0; i<20; i++){
    pthread_join(tid[i], NULL);
    }
    printf("sum = %d\n", sum);
    return 0;
}

在这里插入图片描述
任务二 解决多线程共同访问变量的问题
1、分析任务一,运行结果与预期不符的原因。
2、使用锁机制,解决多线程共同访问变量的问题,所使用的接口为pthread_mutex_lock();
使用方法

先定义并初始化:pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
加锁:pthread_mutex_lock(&mutex);
解锁:pthread_mutex_unlock(&mutex);

3、修改后的代码能够正确预期的结果。
hello.c

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<pthread.h>

pthread_mutex_t mutex;
int sum = 0;

void *thrd_func(void *arg) {
    int i;
    pthread_mutex_lock(&mutex);
    for(i=0; i<10000; i++) {
        sum++;
    }
    pthread_mutex_unlock(&mutex);
}

int main(){
    int ret,i;
    pthread_t tid[20];
    pthread_mutex_init(&mutex,NULL);
    for(i=0; i<20; i++){
    ret = pthread_create(&tid[i], NULL,thrd_func, NULL);
    }

    for(i=0; i<20; i++){
    pthread_join(tid[i], NULL);
    }
    pthread_mutex_destroy(&mutex);
    printf("sum = %d\n", sum);

    return 0;
}

在这里插入图片描述
任务三 使用信号量解决生产者消费者问题
编写程序,模拟生产者消费者的流程。实现思路如下:
1、分别创建N个生产者,N个消费者线程,并创建一个容纳M个商品的缓冲池(可以实现为一个数组,数组元素初始化为0,表示一个空缓冲池)。
2、生产者每次生产一个产品,将产品按顺序(从数组的起始位置顺序存放)放入到缓冲池中(修改数组元素值为1)。
3、消费者每次按顺序(从数组的起始位置顺序取产品)取产品消费(将对应位置的元素值修改为0)。
4、生产者、消费者每次生产/消费完产品之后,睡眠一个随机的时间(0-5秒之间的随机数)。
5、每次生产者/消费者修改了缓冲池的内容后,都将当前缓冲区的内容显示出来。
提示
以某一个生产者线程为例的伪代码:
while(1){
请求空缓冲区信号量;
请求访问缓冲区的互斥锁;
生产一个产品(修改in指针指向的数组值为1)
in指针+1;
打印缓冲区中所有的数据内容;
释放互斥锁;
释放信号量;
sleep(0-5之间的随机数)
}
hello.c

#include<stdio.h>
#include<pthread.h>
#include<stdlib.h>
#include<unistd.h>
#include<semaphore.h>
#define N 10
#define M 10

sem_t p_size;
sem_t c_size;
int buffer[M];
int p_index = 0;
int c_index = 0;

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

void printBuffer() {
	int i;
	for (i = 0;i < M; i++) {
		printf("%d ", buffer[i]);
	}
	printf("\n");
}

void *producer(void* arg) {
	while (1) {
		sem_wait(&p_size);
		pthread_mutex_lock(&mutex);
		buffer[p_index] = 1;
		p_index = (p_index+1) % M;
		printf("producer: ");
		printBuffer();
		pthread_mutex_unlock(&mutex);
		sem_post(&c_size);
		sleep(rand() % 6);
	}
}

void *consumer(void* arg) {
	while (1) {
		sem_wait(&c_size);
		pthread_mutex_lock(&mutex);
		buffer[c_index] = 0;
		c_index = (c_index+1) % M;
		printf("consumer: ");
		printBuffer();
		pthread_mutex_unlock(&mutex);
		sem_post(&p_size);
		sleep(rand() % 6);
	}
}


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

	pthread_t p[N];
	pthread_t c[N];
	
	sem_init(&p_size, 0, M);
	sem_init(&c_size, 0, 0);

	int i;
	for (i = 0;i < N; i++) {
		if (pthread_create(&p[i], NULL, &producer, NULL) != 0) {
			perror("pthread_create");
		}
		if (pthread_create(&c[i], NULL, &consumer, NULL) != 0) {
			perror("pthread_create");
		}
	}

	for (i = 0;i < N; i++) {
		if (pthread_join(p[i], NULL) != 0) {
			perror("pthread_join");
		}
		if (pthread_join(c[i], NULL) != 0) {
			perror("pthread_join");
		}
	}

    return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值