实验3:生产者消费者问题实践(编程实验)

2023-05-16

第1关:生产者消费者问题实践

任务要求:
生产者—消费者之间设置一个具有n个缓存区的缓冲池,生产者进程将他所生产的产品放入一个缓冲;消费者进程可以从一个缓冲区中取走产品去消费。老板不允许消费者进程到一个空缓冲去取产品。老板不允许生产者进程向一个已装满产品且尚未取走的缓冲区投放产品。
使用多线程实现生产者和消费者问题模型,使用锁和信号量控制线程之间的同步。

为了完成本关任务,你需要掌握:1.多线程相关的系统调用,2.使用锁控制进程互斥,3.使用信号量控制进程同步

我要说的:

此时确实也就确实是在要求我们要掌握这方面的知识而不能再一知半解下去了emmm…
掰扯掰扯之前学的点点东西:
pthread_create创建进程的函数
pthread_join 线程等待
偷来的原话:pthread_create调用成功以后,新线程和老线程谁先执行,谁后执行用户是不知道的,这一块取决与操作系统对线程的调度,如果我们需要等待指定线程结束,需要使用pthread_join函数

结合上次的:
*tidppthread_t类型,也就是指向线程标识符的指针
这里: pthread_t th //th是要等待结束的线程的标识
也没什么错,是线程的标识

void *thread_return //指针thread_return指向的位置存放的是终止线程的返回状态。
也就可以pthread_join(th,&thread_return);

使用锁控制进程互斥

在主线程中初始化锁为解锁状态
pthread_mutex_t mutex;
pthread_mutex_init(&mutex, NULL);

访问对象时的加锁操作与解锁操作
加锁 pthread_mutex_lock(&mutex)
释放锁 pthread_mutex_unlock(&mutex)
从其他地方查的解释:互斥锁用于上锁,条件变量用于等待。
还是不太清楚。

利用信号量实现进程同步

先引入头文件 #include <semaphore.h>
初始化信号量: int sem_init(sem_t \*sem, int pshared, unsigned int value);
成功返回0,失败返回-1
参数
sem:指向信号量结构的一个指针
pshared: 不是0的时候,该信号量在进程间共享,否则只能为当前进程的所有线程们共享
value:信号量的初始值
信号量减1操作,当sem=0的时候该函数会堵塞 int sem_wait(sem_t *sem);
成功返回0,失败返回-1
参数
sem:指向信号量的一个指针
信号量加1操作 int sem_post(sem_t *sem);
参数与返回同上
销毁信号量 int sem_destroy(sem_t *sem);
参数与返回同上

不过不得不说还是没怎么学到太多干货啊emmm…

上答案:

void *Consumer()
{
	int nex=0;
    for(int i=0;i<10;i++)
    {
        int time = rand()%10+1;
        pthread_mutex_lock(&mutex);
		nex = nex + 1;
		buffer[in] = nex;
		printf("Consume one message:%d\n", nex);
		fflush(stdout);//printf后请一定调用这句刷新输出缓存
		in = (in + 1) % SIZE;
		pthread_mutex_unlock(&mutex);         //互斥锁解锁
		sem_post(&full); 
    }
}

顺便看看它的主函数:

int main()
{	
    sem_init(&empty, 0, 10);    //信号量初始化(最多容纳10条消息,容纳了10条生产者将不会生产消息)
	sem_init(&full, 0, 0);		
	pthread_mutex_init(&mutex, NULL);  //互斥锁初始化		
	pthread_t producid;	
	pthread_t consumid;		
	pthread_create(&producid, NULL, Producer, NULL);   //创建生产者线程	
	pthread_create(&consumid, NULL, Consumer, NULL);   //创建消费者线程	
	pthread_join(producid, NULL); 	
	pthread_join(consumid, NULL);   
	sem_destroy(&empty);         //信号量的销毁
	sem_destroy(&full);    
	pthread_mutex_destroy(&mutex);   //互斥锁的销毁
	return 0;
}

第2关:进程互斥和同步

任务描述:
桌上有个能盛的下五个水果的空盘子。爸爸不停的向盘中放苹果或桔子,儿子不停的从盘中取出桔子享用,女儿不停的从盘中取出苹果享用。规定三人不能同时从盘中取放水果。试用信号量实现爸爸、儿子和女儿这三个进程之间的同步

这个有点凶,里面有个wait()函数
去偷了波,函数功能:父进程一旦调用了wait就立即阻塞自己,由wait自动分析是否当前进程的某个子进程已经退出,如果让它找到了这样一个已经变成僵尸的子进程,wait就会收集这个子进程的信息,并把它彻底销毁后返回;如果没有找到这样一个子进程,wait就会一直阻塞在这里,直到有一个出现为止。
wait()函数相关说明链接

上答案:

void *Son()
{
	while(1)
	{
		int time = rand() % 10 + 1;         
   
		sem_wait(&orange); 
		pthread_mutex_lock(&mutex);
		printf("儿子取了一个桔子\n") ;
        fflush(stdout);
		pthread_mutex_unlock(&mutex);         //互斥锁解锁
		sem_post(&empty);
	}
}

来个主函数:

int main()
{			
    sem_init(&empty, 0, 5);    //信号量初始化
	sem_init(&orange, 0, 0);
	sem_init(&apple, 0, 0);		
	pthread_mutex_init(&mutex, NULL);  //互斥锁初始化		
	pthread_t dadid;	
	pthread_t daughterid;
	pthread_t sonid;		
	pthread_create(&dadid, NULL, Dad, NULL);   //创建爸爸线程	
	pthread_create(&daughterid, NULL, Daughter, NULL);   //创建女儿线程
	pthread_create(&sonid, NULL, Son, NULL);   //创建儿子线程		
	pthread_join(daughterid, NULL);
	pthread_join(sonid, NULL);   
	
	sem_destroy(&empty);         //信号量的销毁
	sem_destroy(&apple);
	sem_destroy(&orange);    

	

	pthread_mutex_destroy(&mutex);   //互斥锁的销毁

	

	return 0;
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

实验3:生产者消费者问题实践(编程实验) 的相关文章

  • 【shell】用shell脚本判断解析域名返回值是否为200

    bin bash for domain name in 96 cat mnt siteList txt 96 do result num 61 96 curl I s m 10 domain name grep HTTP awk 39 pr
  • Ubuntu实现ssh远程登陆

    目录 1 输入netstat查看是否有ssh的22号端口开放 2 输入 sudo apt get install openssh server 进行ssh服务的安装 3 安装完成后输入 service sshd start 启动ssh服务
  • 运用C++查找素数

    查找素数是在学习C C 43 43 中基本的问题 xff0c 主要是考察对循环的应用 xff0c 逻辑上并不是很难 对于常规的素数查找法 xff0c 解题步骤通常是 xff1a xff08 以查找100以内的素数为例 xff09 1 从2开
  • ARM------->第七天,ADC模块、Qt串口控制助手

    1 ADC 模拟量转换为数字量 模拟量 xff1a 连续的 比如 xff0c 温度 电压 电阻 高度 湿度 数字量 xff1a 离散的 在前几天的代码基础下 xff0c 添加adc c文件 adc h文件 xff0c 修改makefile工
  • 学习夹子入门第一篇《了解pancakeswap 路由》

    内部功能 function sortTokens address tokenA address tokenB internal pure returns address token0 address token1 对令牌地址进行排序 fun
  • Linux开启Docker远程访问并设置安全访问(证书密钥),附一份小白一键设置脚本哦!

    前言 喜欢折腾慢慢看 xff0c 不喜欢折腾直接跳到小简下文的一键脚本那里 xff0c 两分钟搞好 我的博客 xff1a https blog ideaopen cn 我的公众号 xff1a 小简聊开发 开启远程访问 编辑docker se
  • IntelliJ IDEA切换Git远程分支 提交代码 拉取最新代码

    前言 xff1a 众所周知Git用命令提交代码比较繁琐 xff0c 而且还得记住命令 xff0c 那如何让我们更加方便的提交代码呢 以及获取最新内容拉到本地 还有更换你要提交到的分支呢 下面废话不多说 目录 IDEA里切换Git分支 IDE
  • 离散数学在计算机相关领域的应用

    离散数学在数据库中的应用 数据库技术被广泛应用于社会各个领域 xff0c 关系数据库已经成为数据库的主流 xff0c 离散数学中的笛卡儿积是一个纯数学理论 xff0c 是研究关系数据库的一种重要方法 xff0c 显示出不可替代的作用 不仅为
  • 【深入浅出Spring6】第一期——入门

    一 Spring 引言 x1f314 1 准备工作 xff08 1 xff09 创建一个空项目 Spring6 xff0c 为其配置JDK和Maven xff08 2 xff09 创建一个字模块 spring6 001 revelation
  • Spring框架简单介绍

    1 Spring框架的概述 xff1a Spring是一个开源代码的设计层面框架 xff0c 解决的是业务逻辑和其他各层次的松耦合问题 xff0c 主要的思想是面向接口编程 Spring的核心是控制权反转 xff08 IOC xff09 和
  • Android中的页面跳转详解

    Android页面跳转方式 xff1a 1 通过class跳转 Intent intent 61 new Intent 当前Activity xff0c 目标Activity intent setClass MainActivity thi
  • 基数排序的代码

    基数排序就是在桶排序的基础上进行的 xff0c 先比较每个元素的个位 xff0c 在比较十位 一次往上 一共设置0 9 10个桶 xff0c 将符合位数的元素放到相应桶中 xff0c 然后每次比较完一位后 xff0c 在将整个数组重新复制给
  • SpringMVC的响应,SpringMVC上传文件

    数据处理和跳转 xff1a 在前面的博客中 xff0c 我们介绍到SpringMVC的架构是通过前端控制器dispatchservlet将前端传递的请求发送到处理器映射器 xff0c 处理器映射器再将请求返回前端控制器 xff0c 然后让由
  • 多线程并发笔记整理

    1 线程本质就是栈结构 2 进程从规模上要大于线程 xff0c 进程是包含线程的 xff0c 主方法就是主线程 3 线程在创立以后 xff0c 各自之间的执行是互不干扰的 4 创建线程之后 xff0c 如果没有start xff0c 线程是
  • 如何实现多线程减少上下文切换

    多线程处理 xff0c 虽然减少了CPU的浪费 xff0c 但是 xff0c 线程间的切换会导致开销增大 xff0c 如何减少线程的切换 1 无锁并发编程 xff1a 多线程竞争锁时 xff0c 会引起上下文切换 xff0c 所以多线程处理
  • 产生死锁和避免死锁

    如何造成死锁 xff1a 在下面的代码中 xff0c 线程t1对A进行加锁 xff0c 线程t2对B进行加锁 xff0c 但是t1想要B xff0c t2想要A xff0c 这样就会导致 xff0c 两个线程都在等待对方释放锁 xff0c
  • 多线程并发编程中的锁

    1 volatile xff0c 修饰的a在1线程执行完后 xff0c 写回内存刷新了 xff0c 此时缓存行中的线程2所用的a就被标记为无效了 xff0c 要再次从内存读取 xff0c 线程1刷新以后的a 但是他只能保证当前查看的时候是正
  • 消息队列解耦合

    队列 xff1a 传统的串行化服务的缺点是 1 耦合性太强 xff0c xff08 如果发生网络波荡 xff0c 就会导致都失败 xff09 2系统吞吐量不大 xff0c 耗时多 传统的串行化服务的优点是 xff1a 系统结构简单 xff0
  • java内存模型(JMM)

    1 并发编程的两大问题 xff1a 多线程之间如何通信 以及多线程之间如何同步 2 线程之间的通信机制包含 xff1a 共享内存和消息传递 3 保证现成的安全是在承接上次线程写完之后再读 4 xff08 面试点 xff09 java线程的通
  • JavaScript的传参问题,出现bug

    记录一次javascript传参遇到的问题 xff1a javascript是弱编辑语言 xff0c 所以传递参数的时候只需要 xff0c 将参数变量写入到方法中即可 xff0c 但是今天我在使用的时候却发现 xff0c 同一个数组数据中

随机推荐

  • 在测试VPN时候的惊天大坑,命令行查到的ip与百度搜索的ip不一致

    为什么百度查到的IP和ipconfig命令的结果不一样 详解公网IP 私网IP 网络分类 xff08 A B C xff09 蒲公英云 推荐 xff1a Java网络编程汇总 Java 原文地址 Link 1 96 IP 96 可以分为 9
  • Android进度条ProgressBar使用详解

    先介绍一下ProgressBar几种比较常用的属性 布局中设置 xff1a android max 61 34 100 34 最大显示进度 android progress 61 34 50 34 第一显示进度 android second
  • 如何让IDEA连接上你的GitHub---git协作开发的出发站

    1 在你的idea中 xff0c 打开设置 xff0c 然后去到VerSion Control 中找到GitHub xff0c 左上角有一个添加按钮 2 由于GitHub收网络影响不太稳定 xff0c 为了更好连接 xff0c 选择使用口令
  • 火狐浏览器,访问腾讯云服务器的时候,出现建立安全连接失败的问题。

    在腾讯云上部署了一个服务器 xff0c 听过服务器ip访问项目的时候 xff0c 刚开始的时候 xff0c 启动tomcat xff0c 是可以启动的 xff0c 有进程的 xff0c 但是访问的时候 xff0c 会出现超时连接 后来想了想
  • 最右的一道面试算法题,--特殊基因

    题目描述 小右发现某种特殊基因片段可以使人类拥有某种超能力 xff0c 比如飞翔 xff0c 隐身 xff0c 时光倒流等等 他想找到拥有这些超能力的人 一个人要想拥有超能力当且仅当他的基因编码里包含至少一个特殊基因片段作为子串 请编写程序
  • 布隆过滤器原理和仿写

    1 作用 布隆过滤器是一个防止黑客恶意攻击的宝器 xff0c 布隆过滤器可以与redis结合使用 xff0c 能够有效地防止redis缓存穿透 先将全数据 xff0c 存放到过滤器中 当黑客访问时 xff0c 会携带访问数据 xff0c 去
  • 多线程测试力扣练习题

    1115 交替打印FooBar 给定一个类 class FooBar public void foo for int i 61 0 i lt n i 43 43 print 34 foo 34 public void bar for int
  • 2021-09-12

  • 2021-09-12

  • 2021-10-14

  • 【无标题】

    学生成绩管理
  • 【无标题】

    零售业管理
  • Android中ScrollView使用详解

    滚动视图 xff08 ScrollView xff09 是指当拥有很多内容 xff0c 屏幕显示不完时 xff0c 需要通过滚动来显示完整的视图 包括水平滚动视图 xff08 HorizontalScrollView xff09 和垂直滚动
  • 【JAVA】快速排序

    快排 xff0c 和冒泡排序一样 xff0c 是同一类型的排序 xff0c 都是交换排序 交换 xff0c 涉及在遍历中比较 xff0c 然后相互交换元素 冒泡排序是根据趟数两两比较 xff0c 边比较边交换 xff0c 快排也一样 xff
  • 二叉树C语言构建及功能实现

    关于二叉树 xff0c 出现的印象一般就是一个 丫 字 xff0c 这看似没有毛病 xff0c 但其实也真没有什么毛病 xff0c 逻辑上的二叉树差不多就是这个样子 可是当我们已经在逻辑上构建好了二叉树之后 xff0c 真的不想去动手实现实
  • Spring项目的创建和使用

    目录 Spring项目的创建和使用 1 创建Spring项目 1 1基于maven创建项目1 2 在maven项目中添加Spring核心包 1 3 创建一个启动类 2 将对象存储到Spring中 2 1创建一个业务对象 2 2将业务对象储存
  • 二叉树创建的两种方法(图解)

    目录 一 括号表示法 xff08 1 xff09 括号表示法构建二叉树的算法思路及算法实现 xff08 2 xff09 图解括号表示法构建二叉树 xff08 3 xff09 测试程序 二 扩展二叉树 xff08 1 xff09 扩展二叉树构
  • 【嵌入式Linux】手机连接Linux系统的adb操作

    目录 手机连接Linux系统的adb操作过程adb控制指令测试 手机连接Linux系统的adb操作过程 通过了解 xff0c 安卓的底层也是Linux xff0c 这样我们就可以通过Linux控制安卓手机完成一系列指令 xff0c 在这里需
  • GUI(图形用户界面)之事件处理

    Swing组件中的事件处理专门用于响应用户的操作 xff0c 例如 xff0c 响应用户的鼠标单击 按下键盘等操作 在Swing事件处理的过程中 xff0c 主要涉及三类对象 xff1a 1 事件源 xff1a 事件发生的场所 xff0c
  • 实验3:生产者消费者问题实践(编程实验)

    第1关 xff1a 生产者消费者问题实践 任务要求 xff1a 生产者 消费者之间设置一个具有n个缓存区的缓冲池 xff0c 生产者进程将他所生产的产品放入一个缓冲 xff1b 消费者进程可以从一个缓冲区中取走产品去消费 老板不允许消费者进