FreeRTOS深度解析:队列集(Queue Sets)的原理与应用

FreeRTOS深度解析:队列集(Queue Sets)的原理与应用

什么是队列集?

在FreeRTOS中,队列集(Queue Sets,英文名xQueueSet)是一种强大的数据结构,用于高效管理多个队列。它的主要作用是让任务能够同时等待多个队列中的消息,而不必单独轮询每个队列。

队列集本质上是多个队列的集合,提供了一种便捷的方式来监视多个队列,只要集合中的任何一个队列收到消息,任务就能被唤醒并处理该消息。

为什么需要队列集?

在多任务系统中,经常会出现一个任务需要从多个队列接收数据的情况。让我们通过一个实际场景来理解队列集的必要性:

假设我们有一个温湿度监控系统:

  • 任务1负责采集数据:将温度数据发送到队列A,将湿度数据发送到队列B
  • 任务2需要处理这两种数据:必须从队列A和队列B都读取数据

传统方法的问题

如果不使用队列集,任务2需要分别读取两个队列:

// 传统方法:分别读取两个队列
// 先尝试读取队列A
if(xQueueReceive(queueA, &tempData, portMAX_DELAY) == pdTRUE) {// 处理温度数据
}// 再尝试读取队列B
if(xQueueReceive(queueB, &humidData, portMAX_DELAY) == pdTRUE) {// 处理湿度数据
}

这种方法存在严重问题:

  1. 阻塞问题:如果队列A没有数据,任务会阻塞在第一个xQueueReceive调用,即使队列B中已有数据也无法处理
  2. 效率低下:需要编写复杂的轮询逻辑以避免阻塞
  3. 实时性差:无法保证及时处理所有可用数据

队列集的解决方案

队列集优雅地解决了上述问题。它将多个队列组合成一个整体,任务只需监听这个队列集:

// 创建队列集
QueueSetHandle_t queueSet = xQueueCreateSet(totalQueueSize);// 将队列添加到队列集
xQueueAddToSet(queueA, queueSet);
xQueueAddToSet(queueB, queueSet);// 在任务中等待任意队列的数据
QueueSetMemberHandle_t activeMember = xQueueSelectFromSet(queueSet, portMAX_DELAY);// 判断是哪个队列收到了数据
if(activeMember == queueA) {xQueueReceive(queueA, &tempData, 0);// 处理温度数据
} else if(activeMember == queueB) {xQueueReceive(queueB, &humidData, 0);// 处理湿度数据
}

队列集的核心API函数

FreeRTOS提供了几个关键函数来操作队列集:

  1. xQueueCreateSet() - 创建一个新的队列集

    QueueSetHandle_t xQueueCreateSet(const UBaseType_t uxEventQueueLength);
    
  2. xQueueAddToSet() - 将队列添加到队列集

    BaseType_t xQueueAddToSet(QueueHandle_t xQueueOrSemaphore, QueueSetHandle_t xQueueSet);
    
  3. xQueueRemoveFromSet() - 从队列集中移除队列

    BaseType_t xQueueRemoveFromSet(QueueHandle_t xQueueOrSemaphore, QueueSetHandle_t xQueueSet);
    
  4. xQueueSelectFromSet() - 等待并返回队列集中有消息的队列

    QueueSetMemberHandle_t xQueueSelectFromSet(QueueSetHandle_t xQueueSet, TickType_t xTicksToWait);
    
  5. xQueueSelectFromSetFromISR() - 中断服务程序中使用的版本

    QueueSetMemberHandle_t xQueueSelectFromSetFromISR(QueueSetHandle_t xQueueSet);
    

队列集的高级应用案例

队列集不仅可以管理队列,还可以管理信号量,让我们看一个更复杂的应用场景:

假设我们有一个IoT设备,需要同时处理:

  • UART接收的数据(通过队列)
  • 定时器触发的采样任务(通过二值信号量)
  • 外部中断事件(通过计数信号量)

使用队列集可以优雅地实现这一需求:

// 创建资源
QueueHandle_t uartQueue = xQueueCreate(10, sizeof(uint8_t));
SemaphoreHandle_t timerSem = xSemaphoreCreateBinary();
SemaphoreHandle_t extIntSem = xSemaphoreCreateCounting(10, 0);// 创建队列集
QueueSetHandle_t iotQueueSet = xQueueCreateSet(10 + 1 + 10);// 添加到队列集
xQueueAddToSet(uartQueue, iotQueueSet);
xQueueAddToSet(timerSem, iotQueueSet);
xQueueAddToSet(extIntSem, iotQueueSet);// 主任务处理
while(1) {QueueSetMemberHandle_t activeMember = xQueueSelectFromSet(iotQueueSet, portMAX_DELAY);if(activeMember == uartQueue) {// 处理UART数据uint8_t data;xQueueReceive(uartQueue, &data, 0);processUartData(data);} else if(activeMember == timerSem) {// 处理定时事件xSemaphoreTake(timerSem, 0);performScheduledSampling();}else if(activeMember == extIntSem) {// 处理外部中断xSemaphoreTake(extIntSem, 0);handleExternalInterrupt();}
}

使用队列集的注意事项

  1. 队列集中的所有队列/信号量必须为空才能被添加到队列集中
  2. 队列集的大小必须能够容纳所有成员队列/信号量的总容量
  3. 不支持递归互斥量(Recursive Mutex)
  4. 在读取到队列集中有活动的队列后,仍需调用相应的xQueueReceivexSemaphoreTake函数获取实际数据

总结

队列集是FreeRTOS中一个极其实用的功能,能有效提高代码效率和系统响应性。它通过允许任务同时等待多个事件源,减少了代码复杂度,避免了常见的阻塞陷阱,是开发复杂实时系统的得力助手。

如果想深入学习FreeRTOS和队列集的更多高级用法,推荐访问我的GitHub仓库:https://github.com/Despacito0o/FreeRTOS,这里有从入门到精通的全面FreeRTOS学习资源,包括详细的示例项目和中英双语文档!


如果您喜欢这篇文章,欢迎点赞、收藏和关注,您的支持是我创作的最大动力!有任何问题也欢迎在评论区留言交流!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/76890.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

QT creater和vs2017文件路径问题

1. \\双反斜杠,传统写法,需转义 在 C/C 字符串中,\ 具有特殊含义,例如: \n 表示换行 \t 表示制表符 \" 表示双引号 如果要表示一个真正的反斜杠,必须写成 \\,否则编译器会将其解释为转…

对流对象的理解

在c里,“流”可以理解为数据传输与操作的“介质”。 从输入输出角度来看,有输入流(比如cin)和输出流(cout)。对于输入流,数据通过它从外部设备(例如键盘)“流入”程序内…

Visium HD多样本拼片拆分

Visium HD实验的时候一个捕获区域内可以包含多个样本拼片(例如多个组织切片或不同样本的排列)是常见的实验设计,多样本拼片能够提升实验效率,单张玻片处理多个样本,降低试剂和测序成本,后续分析的时候只需要…

进程(Process)详解

进程(Process)详解 一、基本定义 ‌概念‌ 进程是计算机中程序的一次动态执行实例,包含程序代码、数据及运行状态,是操作系统进行资源分配和调度的基本单位‌。与静态的“程序”不同,进程是动态实体,随程…

毕业论文超清pdf带标签导出

Word直接导出的pdf不够清晰,使用打印导出的pdf又不带书签以及目录跳转功能这一问题,查阅网上资料使用Adobe DC似乎能够解决但是下载安装比较麻烦,于是写了python程序解决该问题。 解决思路: 使用python脚本对两个pdf文件进行合并…

NOIP2012提高组.同余方程

目录 题目算法标签: 数论, 扩展欧几里得算法思路代码 题目 203. 同余方程 算法标签: 数论, 扩展欧几里得算法 思路 简单的扩展欧几里得算法应用题, 扩展欧几里得算法可以直接计算同余方程的通解, 因为求得是最小正整数解, 因此需要取模转换为正整数 a x b y ≡ 1 ax by …

C++学习-入门到精通-【0】计算机和C++简介

C学习-入门到精通-[0]计算机和C简介 计算机和C简介 C学习-入门到精通-[0]计算机和C简介一、计算机的组成二、硬件和软件三、数据的层次结构四、机器语言、汇编语言和高级语言五、C标准库六、面向对象技术 一、计算机的组成 计算机是由多个不同功能的逻辑单元组成的&#xff1a…

macOS 系统设置息屏情况下,PHP等后台脚本继续执行

在 macOS 系统下,当屏幕息屏或合上盖子时,后台脚本程序是否会继续运行,主要取决于以下几个因素: 1. 系统睡眠状态的影响 默认情况:合盖/息屏后,Mac 会进入「睡眠模式」(部分硬件休眠&#xff…

SpringBoot集成ActiveMQ异常处理机制:若未捕获异常,消息会被重新投递

一、问题描述 SpringBoot项目集成AvtiveMQ,作为消息消费者。如果在消费消息的方法中,抛出异常,会产生什么效果? 二、ActiveMQ异常处理机制(AI问答仅供参考) 在Spring Boot项目集成ActiveMQ作为消息消费者…

【Java学习笔记】random的使用

random使用方法 使用说明&#xff1a;返回的是(0<n<1)这个范围中的任意带正号的double值 代码实例 public class helloworld{public static void main(String[] args){System.out.println(Math.random());} }生成0-100中的任意数代码示例 public class Main {public …

(三)垂直分库架构、分布式数据库

文章目录 垂直分库架构/分布式数据库什么是垂直分库架构架构模型优缺点优点缺点 技术案例分布式数据库架构模型优缺点优点缺点 技术案例 垂直分库架构/分布式数据库 什么是垂直分库架构 根据业务的模块划分&#xff0c; 将不同业务的数据放到不同的数据库中。 比如一个电子商城…

数据结构线性表的顺序存储结构

线性表是由零个或多个数据元素组成的有序序列。 特点&#xff1a; 数据元素间是有顺序的&#xff1b; 数据元素的个数是有限的&#xff1b; 一般来说&#xff0c;数据元素的类型是相同的&#xff08;强类型语言&#xff09;。c/c是强类型语言&#xff0c;必须指定数据类型。…

扣子空间试用:生成五一骑行规划+notion文章编写

今天试用了一下扣子空间&#xff0c;正好五一快到了&#xff0c;让它帮忙做了五一骑行规划&#xff0c;效果不赖&#xff01; 生成五一骑行规划 点击前往网站查看效果 prompt 如下&#xff1a; 帮我做一个五一上海骑行规划 要求&#xff1a; - 风景优美 - 人少 - 100km总路程…

最新得物小程序sign签名加密,请求参数解密,响应数据解密逆向分析

点击精选&#xff0c;出现https://app.dewu.com/api/v1/h5/index/fire/index 这个请求 直接搜索sign的话不容易定位 直接搜newAdvForH5就一个&#xff0c;进去再搜sign&#xff0c;打上断点 可以看到t.params就是没有sign的请求参数&#xff0c; 经过Object(a.default)该函数…

在C#串口通信中,一发一收的场景,如何处理不同功能码的帧数据比较合理,代码结构好

在 C# 串口通信的一发一收场景里&#xff0c;处理不同功能码的帧数据可采用以下合理的代码结构&#xff0c;它能让代码更具可读性、可维护性和可扩展性。 实现思路 定义帧结构&#xff1a;创建一个类来表示通信帧&#xff0c;其中包含功能码、数据等信息。功能码处理逻辑&…

【C++】vector扩容缩容

vector扩容缩容 1 扩容 一般来说&#xff0c;主要是重新分配内存 2 缩容 resize 缩小后&#xff0c;vector 的容量&#xff08;capacity()&#xff09;可能保持不变&#xff0c;需要显式调用 shrink_to_fit() 来释放内存。 验证代码&#xff1a; #include <vector>…

java中,线程的执行状态有哪些

在 Java 里&#xff0c;线程有 6 种执行状态&#xff0c;这些状态都在 java.lang.Thread.State 枚举类中被定义。下面为你详细介绍这些状态&#xff1a; 1. NEW&#xff08;新建状态&#xff09; 当你创建了一个 Thread 对象&#xff0c;却还未调用其 start() 方法时&#xf…

MATLAB 控制系统设计与仿真 - 41

鲁棒控制的其他函数 - 回路成型函数 loopsyn 灵敏度问题由鲁棒控制工具箱中的loopsyn就可以直接求解,该函数采用H无穷回路成型算法设计控制器,函数的调用格式为: [K,CL,gamma,info] = loopsyn(G,Gd) % G为受控对象模型% Gd为期望的回路传递函数% K为回路成型控制器模型% C…

查询Hologres或postgresql中的数据

因Hologres使用postgresql的语法.所以两者查询一样. 方案1: import java.sql.Connection; import java.sql.DriverManager; import java.sql.SQLException; import java.util.ArrayList; import java.util.List;/*** 一个使用简单连接池管理PostgreSQL连接的工具类。*/ publi…

OpenBayes 一周速览|EasyControl 高效控制 DiT 架构,助力吉卜力风图像一键生成;TripoSG 单图秒变高保真 3D 模型

公共资源速递 10 个教程&#xff1a; * 一键部署 R1-OneVision * UNO&#xff1a;通用定制化图像生成 * TripoSG&#xff1a;单图秒变高保真 3D * 使用 VASP 进行机器学习力场训练 * InfiniteYou 高保真图像生成 Demo * VenusFactory 蛋白质工程设计平台 * Qwen2.5-0mni…