Toggle navigation
首页
问答
文章
积分商城
专家
专区
更多专区...
文档中心
返回主站
搜索
提问
会员
中心
登录
注册
Kernel
第12天:生产者-消费者问题 [15天rt-thread入门学习]
发布于 2018-06-04 21:34:36 浏览:1638
订阅该版
* 本帖最后由 wolfgang2018 于 2018-6-4 22:17 编辑 * 1、 使用互斥量替换sem_lock,查看程序运行结果 用互斥量替换sem_lock后程序结果是一样的: sem_lock的运行结果: ![01.jpg](/uploads/201806/04/212322dbacpyqo3q5evaea.jpg) ![02.jpg](/uploads/201806/04/212331cxli7nljrzwnl7cn.jpg) 这里修改一个bug: 问题是在第二次运行生产者消费者程序的时候,get值还在累加,会造成消费者进程始终挂起;在break前,或者线程退出前需要将get清零。 ``` ........ /* 生产者生产到10个数目,停止,消费者线程相应停止 */ if (get == 10) { get=0;break; } .............. ``` 至于运行结果中: the consumer[32] exit! 与 the consumer[10] exit! rt_kprintf("the consumer[%d] exit!
"); 有关数值不一样属于正常情况 修改为互斥量的代码如下: ``` /* 使用互斥机制 生产消费者*/ /* 指向线程控制块的指针 */ static rt_thread_t producer_tid_mux = RT_NULL; static rt_thread_t consumer_tid_mux = RT_NULL; struct rt_mutex mutex_lock; /* 生成者线程入口 */ void producer_thread_entry_mut(void *parameter){ int cnt = 0; while (cnt < 10) { rt_sem_take(&sem_empty, RT_WAITING_FOREVER); rt_mutex_take(&mutex_lock, RT_WAITING_FOREVER); array[set % MAXSEM]= cnt + 1; rt_kprintf("the producer_mutex generates a number: %d
", array[set % MAXSEM]); rt_sem_release(&sem_full); /* 发布一个满位 */ set++; rt_mutex_release(&mutex_lock); cnt++; /* 暂停一段时间 */ rt_thread_delay(50); } rt_kprintf("the producer exit!
"); } /* 消费者线程入口 */ void consumer_thread_entry_mut(void *parameter){ rt_uint32_t no; rt_uint32_t sum=0; /* 第n个线程,由入口参数传进来 */ no = (rt_uint32_t)parameter; while (1) { rt_sem_take(&sem_full, RT_WAITING_FOREVER); rt_mutex_take(&mutex_lock, RT_WAITING_FOREVER); sum += array[get % MAXSEM]; rt_kprintf("the consumer[%d] get a number: %d
", (get % MAXSEM), array[get % MAXSEM]); get++; rt_sem_release(&sem_empty); /* 释放一个空位 */ rt_mutex_release(&mutex_lock); /* 暂停一小会时间 */ /* 生产者生产到10个数目,停止,消费者线程相应停止 */ if (get == 10) { get=0;break; } } rt_kprintf("the consumer[%d] sum is %d
", no, sum); rt_kprintf("the consumer[%d] exit!
"); } /* 使用互斥机制 生产消费者*/ /* 指向线程控制块的指针 */ static rt_thread_t producer_tid_mux = RT_NULL; static rt_thread_t consumer_tid_mux = RT_NULL; struct rt_mutex mutex_lock; int mutex_producer_consumer_init(){ rt_err_t result; result = rt_mutex_init(&mutex_lock, "mut_lock", RT_IPC_FLAG_FIFO); rt_sem_init(&sem_empty, "empty", MAXSEM, RT_IPC_FLAG_FIFO); rt_sem_init(&sem_full, "full", 0, RT_IPC_FLAG_FIFO); if (result != RT_EOK) { rt_kprintf("init static \"mut_lock\" failed.
"); return -1; } /* 创建线程1 */ producer_tid_mux = rt_thread_create("prod_mut", producer_thread_entry_mut, RT_NULL, THREAD_STACK_SIZE, THREAD_PRIORITY - 1, THREAD_TIMESLICE); if (producer_tid_mux != RT_NULL) { rt_thread_startup(producer_tid_mux); } /* 创建线程2 */ consumer_tid_mux = rt_thread_create("cons_mut", consumer_thread_entry_mut, RT_NULL, THREAD_STACK_SIZE, THREAD_PRIORITY + 1, THREAD_TIMESLICE); if (consumer_tid_mux != RT_NULL) { rt_thread_startup(consumer_tid_mux); } return 0; } /* 如果设置了RT_SAMPLES_AUTORUN,则加入到初始化线程中自动运行 */ #if defined (RT_SAMPLES_AUTORUN) && defined(RT_USING_COMPONENTS_INIT) INIT_APP_EXPORT(semaphore_producer_consumer_init); INIT_APP_EXPORT(mutex_producer_consumer_init) #endif /* 导出到 msh 命令列表中 */ MSH_CMD_EXPORT(semaphore_producer_consumer_init, producer_consumer sample); MSH_CMD_EXPORT(mutex_producer_consumer_init, producer_consumer_mut sample); ``` 另外本人还将互斥量修改为临界区方式,结果是一样的,不过这三种方式,临界区、互斥量、信号量都是为保证“get、Set以及释放信号量”时不会被其它线程抢占,而改变了相应的生产、消费控制。三者虽然都能保护各线程能正确执行,但是实现的原理不一样,在增加生产者或消费者后就相关线程调度就会有所差异。 [https://www.rt-thread.org/qa/thread-7210-1-1.html](第01天:初识RT-Thread [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7211-1-1.html](第02天:系统启动代码和用户入口 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7261-1-1.html](第03天:线程的创建与删除 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7262-1-1.html](第04天:空闲任务及其钩子函数 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7263-1-1.html](第05天:中断和临界区保护 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7264-1-1.html](第06天:堆的初始化和使用 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7265-1-1.html](第07天:信号量的使用 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7266-1-1.html](第08天:互斥量的使用 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7267-1-1.html](第09天:邮箱的使用 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7268-1-1.html](第10天:消息队列的使用 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7271-1-1.html](第11天:事件的使用 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7273-1-1.html](第13天:生产者-消费者问题 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7274-1-1.html](第14天:软件定时器 [15天rt-thread入门学习]) [https://www.rt-thread.org/qa/thread-7275-1-1.html](第15天:内存池[15天rt-thread入门学习])
查看更多
0
个回答
默认排序
按发布时间排序
暂无答案,快来添加答案吧
撰写答案
登录
注册新账号
关注者
0
被浏览
1.6k
关于作者
wolfgang2018
这家伙很懒,什么也没写!
提问
16
回答
2
被采纳
0
关注TA
发私信
相关问题
1
请教cpu使用率分析
2
选择FreeRTOS, 还是RT-Thread。
3
thread heap stack overflow ?
4
rtt消息队列delay问题
5
释放被删除线程的内存地方在哪里啊
6
请教:各线程结束后,释放其中的内存的连续性问题
7
STM32F103中断关于信号量、邮箱问题
8
RTT中的线程栈大小如何控制
9
关于线程由执行态变为挂起态的代码实现,,,
10
rt_malloc(rt_size_t size)内存分配函数最小分配尺寸问题
推荐文章
1
RT-Thread应用项目汇总
2
玩转RT-Thread系列教程
3
国产MCU移植系列教程汇总,欢迎查看!
4
机器人操作系统 (ROS2) 和 RT-Thread 通信
5
五分钟玩转RT-Thread新社区
6
【技术三千问】之《玩转ART-Pi》,看这篇就够了!干货汇总
7
关于STM32H7开发板上使用SDIO接口驱动SD卡挂载文件系统的问题总结
8
STM32的“GPU”——DMA2D实例详解
9
RT-Thread隐藏的宝藏之completion
10
【ART-PI】RT-Thread 开启RTC 与 Alarm组件
最新文章
1
使用百度AI助手辅助编写一个rt-thread下的ONVIF设备发现功能的功能代码
2
RT-Thread 发布 EtherKit开源以太网硬件!
3
rt-thread使用cherryusb实现虚拟串口
4
《C++20 图形界面程序:速度与渲染效率的双重优化秘籍》
5
《原子操作:程序世界里的“最小魔法单位”解析》
热门标签
RT-Thread Studio
串口
Env
LWIP
SPI
AT
Bootloader
Hardfault
CAN总线
FinSH
ART-Pi
USB
DMA
文件系统
RT-Thread
SCons
RT-Thread Nano
线程
MQTT
STM32
RTC
FAL
rt-smart
ESP8266
I2C_IIC
WIZnet_W5500
ota在线升级
UART
PWM
cubemx
freemodbus
flash
packages_软件包
BSP
潘多拉开发板_Pandora
定时器
ADC
GD32
flashDB
socket
中断
Debug
编译报错
msh
SFUD
keil_MDK
rt_mq_消息队列_msg_queue
at_device
ulog
C++_cpp
本月问答贡献
踩姑娘的小蘑菇
7
个答案
3
次被采纳
张世争
8
个答案
2
次被采纳
rv666
5
个答案
2
次被采纳
a1012112796
13
个答案
1
次被采纳
用户名由3_15位
11
个答案
1
次被采纳
本月文章贡献
程序员阿伟
6
篇文章
2
次点赞
hhart
3
篇文章
4
次点赞
大龄码农
1
篇文章
2
次点赞
ThinkCode
1
篇文章
1
次点赞
Betrayer
1
篇文章
1
次点赞
回到
顶部
发布
问题
分享
好友
手机
浏览
扫码手机浏览
投诉
建议
回到
底部