当前位置: 首页 > news >正文

Linux平台下线程同步,实现“生产者消费者问题”


(1)线程同步,实现生产者消费者问题

要求:缓冲区大小为20,生产者每次放一个产品,消费者每次取走一个产品;生产者和消费者至少2个。

(2)代码如下:

#include <stdio.h>
#include <pthread.h>
#include <unistd.h>
#include <sched.h>
void *producter_f (void *arg);					/*生产者*/
void *consumer_f (void *arg);					/*消费者*/
int buffer_has_item=0;							/*缓冲区计数值*/
pthread_mutex_t mutex;								/*互斥区*/
//pthread_cond_t mqlock= PTHREAD_COND_INITIALIZER;							
int running =1 ;								/*线程运行控制*/
int main (void)
{
	pthread_t consumer_t;						/*消费者线程参数*/
	pthread_t producter_t;						/*生产者线程参数*/
	pthread_mutex_init (&mutex,NULL);			/*初始化互斥*/
	pthread_create(&producter_t, NULL,(void*)producter_f, NULL );														/*建立生产者线程*/
	pthread_create  (&consumer_t, NULL, (void *)consumer_f, NULL);													/*建立消费者线程*/
	//usleep(1);									/*等待,线程创建完毕*/
	//running =0;									/*设置线程退出值*/
	pthread_join(consumer_t,NULL);				/*等待消费者线程退出*/
	pthread_join(producter_t,NULL);			/*等待生产者线程退出*/
	pthread_mutex_destroy(&mutex);				/*销毁互斥*/
	
	return 0;
}
void *producter_f (void *arg)					/*生产者线程程序*/
{
	while(running)								/*没有设置退出值*/
	{	
		pthread_mutex_lock (&mutex);			/*进入互斥区*/
		if(buffer_has_item > 20)                /*缓冲区满则不生产*/
		{
			pthread_mutex_unlock(&mutex);	
			usleep(1);
		}
		buffer_has_item++;						/*增加计数值*/
		printf("生产,总数量:%d\n",buffer_has_item); 	/*打印信息*/
		pthread_mutex_unlock(&mutex);			/*离开互斥区*/
	}
}
void *consumer_f(void *arg)						/*消费者线程程序*/
{
	while(running)								/*没有设置退出值*/
	{
		pthread_mutex_lock(&mutex);				/*进入互斥区*/
		buffer_has_item--;						/*减小计数值*/
		if(buffer_has_item <  0)                /*缓冲区空则不消费*/
		{
			pthread_mutex_unlock(&mutex);	
			usleep(1);
		}
		printf("消费,总数量:%d\n",buffer_has_item);	/*打印信息*/
		pthread_mutex_unlock(&mutex);			/*离开互斥区*/
	}
}
























(2)第二种,加入同步机制


#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <pthread.h>
#include <semaphore.h>
#define N 2   // 消费者或者生产者的数目
#define M 20 // 缓冲数目

int in = 0;   // 生产者放置产品的位置
int out = 0; // 消费者取产品的位置
int buff[M] = { 0 }; // 缓冲初始化为0, 开始时没有产品
sem_t empty_sem; // 同步信号量, 当满了时阻止生产者放产品
sem_t full_sem;   // 同步信号量, 当没产品时阻止消费者消费
pthread_mutex_t mutex; // 互斥信号量, 一次只有一个线程访问缓冲
int product_id = 0;   //生产者id
int prochase_id = 0; //消费者id

/* 打印缓冲情况 */
void print()
{
	int i;
	for (i = 0; i < M; i++)
	{
		printf("%d ", buff[i]);
	}
	printf("\n");
}

/* 生产者方法 */
void *product()
{
	int id = ++product_id; //当生产者进入的时候 执行计数加1
	while (1)
	{
		// 用sleep的数量可以调节生产和消费的速度,便于观察
		sleep(1);
		sem_wait(&empty_sem);
		pthread_mutex_lock(&mutex);
		in = in % M;
		printf("product%d in %d. like: \t", id, in);
		buff[in] = 1;
		print();
		++in;
		pthread_mutex_unlock(&mutex);
		sem_post(&full_sem);
	}
}

/* 消费者方法 */
void *prochase()
{
	int id = ++prochase_id;
	while (1)
	{
		// 用sleep的数量可以调节生产和消费的速度,便于观察
		sleep(1);
		sem_wait(&full_sem);
		pthread_mutex_lock(&mutex);
		out = out % M;
		printf("prochase%d in %d. like: \t", id, out);
		buff[out] = 0;
		print();
		++out;
		pthread_mutex_unlock(&mutex);
		sem_post(&empty_sem);
	}
}

int main()

{
	pthread_t id1[N];
	pthread_t id2[N];
	int i;
	int ret[N];
	// 初始化同步信号量
	int ini1 = sem_init(&empty_sem, 0, M);
	int ini2 = sem_init(&full_sem, 0, 0);
	if (ini1 && ini2 != 0)
	{
		printf("sem init failed \n");
		exit(1);
	}

	//初始化互斥信号量

	int ini3 = pthread_mutex_init(&mutex, NULL);
	if (ini3 != 0)
	{
		printf("mutex init failed \n");
		exit(1);
	}

	// 创建N个生产者线程
	for (i = 0; i < N; i++)
	{
		ret[i] = pthread_create(&id1[i], NULL, product, (void *) (&i));
		if (ret[i] != 0)
		{
			printf("product%d creation failed \n", i);
			exit(1);
		}
	}

	//创建N个消费者线程
	for (i = 0; i < N; i++)
	{
		ret[i] = pthread_create(&id2[i], NULL, prochase, NULL);
		if (ret[i] != 0)
		{
			printf("prochase%d creation failed \n", i);
			exit(1);
		}
	}
	//销毁线程
	for(i=0;i<N;i++)
	{
		pthread_join(id1[i],NULL);
		pthread_join(id2[i],NULL);
	}
	exit(0);
}

运行结果:






相关文章:

  • aaaa
  • Hibernate懒加载策略
  • [redis] Jedis 与 ShardedJedis 设计
  • Ubuntu中使用GCC编译器出现:没有那个文件或目录
  • 使用 nvm 管理不同版本的 node 与 npm
  • 数据库连接失败Communications link failure due to underlying exception!
  • js遍历 子节点 子元素
  • Hinernate进行mysql数据库的逆向工程无法生成主外键关系
  • lintcode 中等题:find the missing number 寻找缺失的数
  • Myeclipse使用DB Browser连接数据库错误:OPTION SQL_SELECT_LIMIT=DEFAULT
  • C语言如何清除scanf()缓存
  • 通过XmlDocument读写Xml文档参考地址
  • Myeclipse使用hibernate的逆向工程
  • brew 更新
  • 安装vmare-tools——实现ubuntu与windows的互相复制与粘贴(无需共享文件夹)
  • ES6指北【2】—— 箭头函数
  • Docker 笔记(1):介绍、镜像、容器及其基本操作
  • ES6 学习笔记(一)let,const和解构赋值
  • Go 语言编译器的 //go: 详解
  • IOS评论框不贴底(ios12新bug)
  • Java精华积累:初学者都应该搞懂的问题
  • JS专题之继承
  • PHP的Ev教程三(Periodic watcher)
  • python 装饰器(一)
  • Python打包系统简单入门
  • vue.js框架原理浅析
  • vue学习系列(二)vue-cli
  • 从0搭建SpringBoot的HelloWorld -- Java版本
  • 大整数乘法-表格法
  • 电商搜索引擎的架构设计和性能优化
  • 构造函数(constructor)与原型链(prototype)关系
  • 浅谈JavaScript的面向对象和它的封装、继承、多态
  • 转载:[译] 内容加速黑科技趣谈
  • 宾利慕尚创始人典藏版国内首秀,2025年前实现全系车型电动化 | 2019上海车展 ...
  • 教程:使用iPhone相机和openCV来完成3D重建(第一部分) ...
  • # Maven错误Error executing Maven
  • $NOIp2018$劝退记
  • (2)STL算法之元素计数
  • (Java数据结构)ArrayList
  • (二)换源+apt-get基础配置+搜狗拼音
  • (全部习题答案)研究生英语读写教程基础级教师用书PDF|| 研究生英语读写教程提高级教师用书PDF
  • (十三)Flask之特殊装饰器详解
  • (十五)devops持续集成开发——jenkins流水线构建策略配置及触发器的使用
  • (原創) 未来三学期想要修的课 (日記)
  • (转)JAVA中的堆栈
  • (转)VC++中ondraw在什么时候调用的
  • (转载)虚函数剖析
  • .bat批处理(十一):替换字符串中包含百分号%的子串
  • .NET CORE 2.0发布后没有 VIEWS视图页面文件
  • .NET Core实战项目之CMS 第十二章 开发篇-Dapper封装CURD及仓储代码生成器实现
  • .NET与java的MVC模式(2):struts2核心工作流程与原理
  • .net中调用windows performance记录性能信息
  • @property @synthesize @dynamic 及相关属性作用探究
  • [ 隧道技术 ] 反弹shell的集中常见方式(四)python反弹shell
  • [8-23]知识梳理:文件系统、Bash基础特性、目录管理、文件管理、文本查看编辑处理...