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

FreeRTOS入门(03):队列、信号量、互斥量

文章目录

  • 目的
  • 队列(queue)
  • 信号量(semaphore)
  • 互斥量(mutex)
    • 互斥量
    • 递归互斥量
  • 总结

目的

FreeRTOS提供给用户最核心的功能是任务(Task),实际项目中通常会有多个任务,任务间多数时候会需要配合工作,这时候就需要用到 队列、信号量、互斥量 等功能了,这篇文章将对相关内容做个介绍。

本文接上篇:《FreeRTOS入门(02):任务基础使用与说明》

队列(queue)

队列是用于任务间传递数据最常用的东西。队列相当于一个缓存,可以向里面写数据,也可以从里面拿数据,遵循先进先出的原则(FIFO),先写入的数据在取的时候也会先取出。

向队列中写数据则队列的可用空间将减少,从队列中读取数据后该数据将从队列中删除,队列的可用空间会增加。如果队列已满的情况下向其中写数据将阻塞直到有空间可以写入;如果从队列中读取时队列为空将阻塞直到有数据可以读取。

读写操作时如果有多个任务都在等待,那么优先级最高的任务将先获得执行权限,如果优先级相同,那么等待时间最久的任务获得执行权限。

队列操作常用的一些函数如下:

// 创建队列,成功的话返回队列句柄,失败则返回NULL
// uxQueueLength为队列长度,uxItemSize为每个数据的容量(单位为字节)
QueueHandle_t xQueueCreate( UBaseType_t uxQueueLength, UBaseType_t uxItemSize );
// 将队列恢复到空的状态
BaseType_t xQueueReset( QueueHandle_t xQueue );
// 删除队列
void vQueueDelete( QueueHandle_t xQueue )// 写入数据到队列(队尾)
// pvItemToQueue为数据起始地址指针,会从这里开始读入uxItemSize字节数据到队列
// xTicksToWait为队满无法写入时的写入操作超时时间(以Tick计,可以使用pdMS_TO_TICKS(ms)将毫秒时间转换成Tick)
// INCLUDE_vTaskSuspend为1时,xTicksToWait为portMAX_DELAY时将无限期阻塞(不会超时)
// 写入成功返回pdTRUE,否则返回errQUEUE_FULL
BaseType_t xQueueSend( QueueHandle_t xQueue, const void * pvItemToQueue, TickType_t xTicksToWait );
// 功能同上
BaseType_t xQueueSendToBack( QueueHandle_t xQueue, const void * pvItemToQueue, TickType_t xTicksToWait );// 写入数据到队列(队尾)用于中断中使用,不会阻塞
// pxHigherPriorityTaskWoken可写NULL
BaseType_t xQueueSendFromISR( QueueHandle_t xQueue, const void *pvItemToQueue, BaseType_t *pxHigherPriorityTaskWoken);
// 功能同上
BaseType_t xQueueSendToBackFromISR( QueueHandle_t xQueue, const void *pvItemToQueue, BaseType_t *pxHigherPriorityTaskWoken);// 写入数据到队列(队首)
// 写数据到队首其实不怎么符合队列本身设计思想的,不过特殊情况下还是有这个需求的
BaseType_t xQueueSendToFront( QueueHandle_t xQueue, const void * pvItemToQueue, TickType_t xTicksToWait );
// 写入数据到队列(队首)用于中断中使用,不会阻塞
BaseType_t xQueueSendToFrontFromISR( QueueHandle_t xQueue, const void *pvItemToQueue, BaseType_t *pxHigherPriorityTaskWoken);// 从队列读取uxItemSize字节数据到pvBuffer,读取完成后队首指针将移动到下一个数据
// 成功返回pdTRUE,否则返回pdFALSE(超时)
BaseType_t xQueueReceive( QueueHandle_t xQueue, void *pvBuffer, TickType_t xTicksToWait );
// 上面函数用于中断中使用的形式,不会阻塞
BaseType_t xQueueReceiveFromISR( QueueHandle_t xQueue, void *pvBuffer, BaseType_t *pxHigherPriorityTaskWoken );// 获取队首数据,并且队首指针不移动
BaseType_t xQueuePeek( QueueHandle_t xQueue, void *pvBuffer, TickType_t xTicksToWait );
// 上面函数用于中断中使用的形式,不会阻塞
BaseType_t xQueuePeekFromISR( QueueHandle_t xQueue, void *pvBuffer );

除了上面列出的一些操作队列的函数,还有一些查询队列以写、可写、队空、队满等功能的函数,如果有需要可以查询官方文档。

#include "debug.h"
#include "FreeRTOS.h" // 引入头文件
#include "task.h"     // 引入头文件
#include "queue.h"     // 引入头文件QueueHandle_t xQueue; // 队列句柄void task1(void *pvParameters) {while(1){int tick = xTaskGetTickCount();xQueueSend(xQueue, &tick, portMAX_DELAY); // 向队列写数据vTaskDelay(500);}
}void task2(void *pvParameters) {while(1){int data;xQueueReceive(xQueue, &data, portMAX_DELAY); // 从队列读取数据printf("Task2 data is %d\r\n", data);}
}int main(void) {NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);SystemCoreClockUpdate();Delay_Init();USART_Printf_Init(115200);xQueue = xQueueCreate(3, 4); // 创建队列,队列长度为3,每个空间容纳4个字节xTaskCreate(task1, "task1", 256, NULL, 5, NULL); // 创建一个任务xTaskCreate(task2, "task2", 256, NULL, 5, NULL); // 创建一个任务vTaskStartScheduler(); // 任务调度,任务将在这里根据情况开始运行,程序将在这里无序循环while(1) {} // 程序不会运行到这里
}

在这里插入图片描述

信号量(semaphore)

队列用于任务间传递数据使用,使用时会占据比较大的空间,而且读写性能稍差。有些时候任务间交互可能并不需要传递数据,只要知道 资源 有还是没有、或者有多少而已,这个时候使用信号量就比较好了。

信号量分为二进制信号量和计数信号量两种。二进制只有两个状态,表示资源有或者没有(也叫做二值信号量);计数信号量可以表达出资源的数量。信号量的值比较常见的是0、1、2、3……,0表示没有资源可用。

写信号量时,如果不可写入也不会阻塞,会直接返回错误码,这是和队列最大的不同。读信号时如果没有资源则会阻塞。

信号量相关操作函数如下:

// 创建二进制信号量
SemaphoreHandle_t xSemaphoreCreateBinary( void )
// 创建计数信号量
// uxMaxCount为最大计数值,uxInitialCount为初始计数值
SemaphoreHandle_t xSemaphoreCreateCounting( UBaseType_t uxMaxCount, UBaseType_t uxInitialCount )// 删除信号量
void vSemaphoreDelete( SemaphoreHandle_t xSemaphore )// 给出资源,信号量的值+1,不会阻塞
// 操作成功返回pdTRUE,否则返回errQUEUE_FULL
xSemaphoreGive( SemaphoreHandle_t xSemaphore );
// 上面函数的中断函数中使用的版本
xSemaphoreGiveFromISR( SemaphoreHandle_t xSemaphore, signed BaseType_t *pxHigherPriorityTaskWoken )// 获取资源,如果没有资源可用(值为0)则会阻塞知道可用,获取后信号量的值-1
// INCLUDE_vTaskSuspend为1时,xTicksToWait为portMAX_DELAY时将无限期阻塞(不会超时)
// 成功返回pdTRUE,否则返回pdFALSE(超时)
xSemaphoreTake( SemaphoreHandle_t xSemaphore, TickType_t xTicksToWait )
// 上面函数的中断函数中使用的版本
xSemaphoreTakeFromISR( SemaphoreHandle_t xSemaphore, signed BaseType_t *pxHigherPriorityTaskWoken )// 返回信号量计数值
UBaseType_t uxSemaphoreGetCount( SemaphoreHandle_t xSemaphore )

下面是信号量使用演示:

#include "debug.h"
#include "FreeRTOS.h" // 引入头文件
#include "task.h"     // 引入头文件
#include "semphr.h"     // 引入头文件SemaphoreHandle_t xSemaphore; // 信号量句柄void task1(void *pvParameters) {while(1){// xSemaphore初始为1,最大为3xSemaphoreGive(xSemaphore); // 给出  +1 = 2xSemaphoreGive(xSemaphore); // 给出  +1 = 3xSemaphoreGive(xSemaphore); // 给出  此处操作将失败xSemaphoreGive(xSemaphore); // 给出  此处操作将失败vTaskDelete(NULL);}
}void task2(void *pvParameters) {while(1){vTaskDelay(500);xSemaphoreTake(xSemaphore, portMAX_DELAY); // 获取printf("Tick: %d\r\n", xTaskGetTickCount()); // 此例中这行最终会打印三次}
}/* 主函数 */
int main(void) {NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);SystemCoreClockUpdate();Delay_Init();USART_Printf_Init(115200);xSemaphore = xSemaphoreCreateCounting(3, 1); // 创建信号量,最大值3,初始值1xTaskCreate(task1, "task1", 256, NULL, 5, NULL); // 创建一个任务xTaskCreate(task2, "task2", 256, NULL, 5, NULL); // 创建一个任务vTaskStartScheduler(); // 任务调度,任务将在这里根据情况开始运行,程序将在这里无序循环while(1) {}// 程序不会运行到这里
}

在这里插入图片描述

互斥量(mutex)

互斥量其实也是一种信号量,特别是普通的互斥量其实和二进制信号量挺像的。需要注意的是 中断函数中不能使用互斥量

互斥量

普通的互斥量,和二进制信号量很像,其它操作函数基本都是通用的(不能在中断中使用)。创建互斥量函数如下:

SemaphoreHandle_t xSemaphoreCreateMutex( void )

互斥量注意是用来对特定的资源(比如全局变量、外设等)进行保护用的。比如某个任务使用串口输出信息时,如果有优先级更高的任务插入进来,也使用这个串口输出信息,那可能最终串口输出的内容可能就会穿插糅合再一起了。这通常是不符合要求的,所以就需要用互斥量了。

互斥量也被叫做互斥锁,某个任务要用资源时先申请互斥量上个锁,用完资源后解锁。锁在被锁上的时候其它任务无法重复上锁,会阻塞到锁被释放后才能成功上锁。使用互斥量时需要注意的是上锁和解锁要成对出现,并且谁上锁谁就必须解锁。

下面是个互斥量使用演示:

#include "debug.h"
#include "FreeRTOS.h" // 引入头文件
#include "task.h"     // 引入头文件
#include "semphr.h"     // 引入头文件SemaphoreHandle_t xMutex; // 互斥量句柄void task(void *pvParameters) {while(1){xSemaphoreTake(xMutex, portMAX_DELAY); // 上锁printf((const char *)pvParameters);xSemaphoreGive(xMutex);  // 解锁vTaskDelay(5);}
}/* 主函数 */
int main(void) {NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);SystemCoreClockUpdate();Delay_Init();USART_Printf_Init(9600); // 9600波特率下每秒约可发送960字节xMutex = xSemaphoreCreateMutex(); // 创建互斥量xTaskCreate(task, "task1", 256, "USART1_Task1: ABCDEFGHIJKLMNOPQRSTUVWXYZ\r\n", 5, NULL); // 创建一个任务xTaskCreate(task, "task2", 256, "USART1_Task2: 01234567890123456789\r\n", 7, NULL); // 创建一个任务 该任务优先级更高,会抢占前面的任务vTaskStartScheduler(); // 任务调度,任务将在这里根据情况开始运行,程序将在这里无序循环while(1) {} // 程序不会运行到这里
}

在这里插入图片描述
上面演示中 task2 有更高的优先级,会抢占 task1 ,所以 task2 有更多的机会运行,也就有更多时间可以使用串口输出信息。但是因为有互斥量存在,所以每个任务在上锁之后使用串口,在解锁前都可以独占串口,不会被抢占,所以最终都能完整的输出信息。

如果注释掉代码中上锁和解锁操作,那么最终输出会变成下面这样:
在这里插入图片描述
仔细看可以发现,因为 task2 有更高优先级,不会被 task1 抢占,所以 task2 可以完整的输出信息。但是 task1 输出过程中就会被 task2 插入打断。

递归互斥量

上面的互斥量有点像二进制信号量,而递归互斥量就有点像计数信号量。递归互斥量可以给多次上锁,但用完后上了几次锁就需要解几次锁。

递归互斥量部分函数也可以和上面公用,但下面几个是它特有的:

SemaphoreHandle_t xSemaphoreCreateRecursiveMutex( void )
xSemaphoreGiveRecursive( SemaphoreHandle_t xMutex )
xSemaphoreTakeRecursive( SemaphoreHandle_t xMutex, TickType_t xTicksToWait );

总结

队列、信号量、互斥量的使用比较简单,这几个也是大多数RTOS都有的功能。

FreeRTOS中用于任务之间配合工作的附加功能除了这几个还有其它一些,比如任务通知、事件组、队列集、流与消息缓冲区等。这些功能可以为项目开发带来更多的便利性,当然没有它们项目需求也能实现。有时间的话会在后续的文章中介绍这些功能。

相关文章:

FreeRTOS入门(03):队列、信号量、互斥量

文章目录目的队列(queue)信号量(semaphore)互斥量(mutex)互斥量递归互斥量总结目的 FreeRTOS提供给用户最核心的功能是任务(Task),实际项目中通常会有多个任务&#xff…...

Biome-BGC在模拟过程中,如何使用Linux、Python等,完成前处理和后处理工作???

在Biome-BGC模型中,对于碳的生物量积累,采用光合酶促反应机理模型计算出每天的初级生产力(GPP),将生长呼吸和维持呼吸减去后的产物分配给叶、枝条、干和根。生物体的碳每天都按一定比例以凋落方式进入凋落物碳库;对于水份输运过程…...

【unittest学习】unittest框架主要功能

1.认识unittest在 Python 中有诸多单元测试框架,如 doctest、unittest、pytest、nose 等,Python 2.1 及其以后的版本已经将 unittest 作为一个标准模块放入 Python 开发包中。2.认识单元测试不用单元测试框架能写单元测试吗?答案是肯定的。单…...

京东测开岗3+1面经+经验分享,拿到offer,月薪34k....

现在,招聘黄金时间已经来临,在网上看了很多大佬的面经,也加了很多交流群,受到了很多朋友的提点,今天终于轮到我来分享面经啦,之前面试了几家公司,最后拿到了京东测试岗的 offer,这里…...

后端接收格式为x-www-form-urlencoded的数据

1.x-www-form-urlencoded是什么? x-www-form-urlencoded纸面翻译即所谓url格式的编码,是post的默认Content-Type,其实就是一种编码格式,类似json也是一种编码传输格式。form表单中使用 form的enctype属性为编码方式&#xff0…...

LeetCode 707. 设计链表

LeetCode 707. 设计链表 难度:middle\color{orange}{middle}middle 题目描述 设计链表的实现。您可以选择使用单链表或双链表。单链表中的节点应该具有两个属性:valvalval 和 nextnextnext。valvalval 是当前节点的值,nextnextnext 是指向下…...

HTTP的主要作用是什么

1、客户与服务器建立连接; 2、客户向服务器提出请求; 3、服务器接受请求,并根据请求返回相应的文件作为应答; 4、客户与服务器关闭连接。 HTTP的性质: 1、HTTP是一种无状态协议,即服务器不保留与客户交…...

SpringBoot系列-- @Enable 模块驱动

Enable 模块驱动 Enable 模块驱动是以 Enable 为前缀的注解驱动编程模型。所谓 “模块” 是指具备相同领域的功能组件集合,组合所形成一个独立的单元。比如 WebMVC 模块、AspectJ 代理模块、Caching (缓存)模块、JMX (Java 管理扩…...

PHP程序员适合创业吗?

创业是一件自然而然的事,不需要人为选择。 只要你是一个努力能干主动的人,当你在一个行业深耕5年之后,就会发现人生发展的下一步就是创业。当然如果行业合适的话。 什么叫行业合适呢? 就是创业的成本并不那么高,不需…...

2023年CDGA考试-第12章-元数据(含答案)

2023年CDGA考试-第12章-元数据(含答案) 单选题 1.元数据架构的类型主要有四种下列哪项不属于分布式元数据架构的优点? A.减少了批处理 B.元数据的质量完全取决于源系统 C.最大程度的减少了实施和维护所需的工作量 D.元数据总是尽可能保持最新且有效 答案 B 2.元数据管理是…...

数据结构之顺序表篇

一、顺序表概念 二、顺序表各类接口实现 *顺序表初始化 **顺序表销毁 ***顺序表插入操作 ****顺序表删除操作 *****顺序表查找操作 ******顺序表实现打印操作 三、顺序表整体实现源码 *SeqList.h **SeqList.c ***test.c 一、顺序表概念 讲顺序表之前先引入线性表概念&#xff…...

ZBC通证月内已翻倍,Nautilus Chain 上线前夕的“开门红”

近日,Zebec Protocol生态通证ZBC迎来了大涨,据悉该通证月内最高涨幅接近了100%,为一众投资者、社区用户、Zepoch节点等带来了可观的回报,并为生态发展注入了十足的信心。我们看到,Zebec Protocol生态在近期宣布了“销毁…...

人工智能练习题:激活函数需要满足的条件、提高CNN的泛化能力、CNN输出特征图大小计算

文章目录1.激活函数需要满足的条件2.提高CNN泛化能力的方法3.CNN输出特征图大小计算第一次用ChatGPT,不得不说在处理大学生作业上,ChatGPT比国内的作业软件好用多了(感叹)。 1.激活函数需要满足的条件 通常情况下,激活…...

KingbaseES Json 系列三:Json数据操作函数一

KingbaseES Json 系列三--Json数据操作函数一(JSONB_EACH,JSONB_EACH_TEXT,JSONB_OBJECT_KEYS,JSONB_EXTRACT_PATH,JSONB_EXTRACT_PATH_TEXT,JSON_EACH,JSON_EACH_TEXT,JSON_OBJECT_KEYS,JSON_EXTRACT_PATH,JSON_EXTRACT_PATH_TEXT) JSON 数据类型是用来存储 JSON(JavaScript O…...

《设计模式》单例模式

《设计模式》单例模式 单例模式是一种常用的设计模式,其主要优点有: 提供了对唯一实例的全局访问。单例模式保证了整个系统中只有一个实例,这样就可以方便地对该实例进行访问和操作,避免了多个实例之间的冲突和不一致。避免了重…...

C/C++每日一练(20230224)

目录 1. 字符串排序 2. Excel表列名称 3. 颠倒二进制位 附录&#xff1a; 位移运算符 左移运算符<< 1.无符号 2.有符号 右移运算符>> 1.无符号 2.有符号 程序测试 1. 字符串排序 编写程序&#xff0c;输入若干个字符串。 要求: &#xff08;1&#x…...

基于YOLO的酸枣病虫害检测识别实践

在我前面的博文中对于农作物病虫害的检测识别已经做过了&#xff0c;不过那个主要是针对水稻的&#xff0c;文章如下&#xff1a;《基于yolov5的轻量级水稻虫害目标检测项目实践》感兴趣的话可以自行移步阅读。这里主要是针对酸枣常见的几种病虫害检测检测识别&#xff0c;首先…...

WAF:ModSecurity on Nginx(15)

预备知识 Nginx概述 Nginx ("engine x") 是一个高性能的HTTP和 反向代理 服务器&#xff0c;也是一个 IMAP/POP3/SMTP服务器。 Nginx 是由 Igor Sysoev 为俄罗斯访问量第二的 Rambler.ru 站点开发的&#xff0c;第一个公开版本0.1.0发布于2004年10月4日。其将源代…...

Qt 第3课、Qt 中的字符串类

1、C 标准库 STL STL 是意义上需要与C 一同发布的标准库STL 是一套以模板技术完成的 C类库STL 中包含了常用的算法和数据结构STL 包含了字符串类 2、Qt 和 STL STL 的具体实现依赖于编译器生产厂商STL 的 “标准” 只是其接口是标准的 — 相同的全局函数 — 相同的算法类和数…...

Vulnhub靶场----6、DC-6

文章目录一、环境搭建二、渗透流程三、思路总结一、环境搭建 DC-6下载地址&#xff1a;https://download.vulnhub.com/dc/DC-6.zip kali&#xff1a;192.168.144.148 DC-6&#xff1a;192.168.144.154 靶机描述&#xff1a;选择带k01的密码后面会用到 访问192.168.144.154&…...

华为OD机试真题Python实现【去重求和】真题+解题思路+代码(20222023)

去重求和 给定一个数组,编写一个函数, 计算他的最大N个数和最小N个数的和, 需要对数组进行去重。 🔥🔥🔥🔥🔥👉👉👉👉👉👉 华为OD机试(Python)真题目录汇总 输入 第一行输入M,M表示数组大小 第二行输入M个数,表示数组内容 第三行输入N表示需要…...

lammps教程:Ovito选择特定晶粒的方法

大家好&#xff0c;我是小马老师。 本文介绍如何使用ovito提取特定的晶粒。 在多晶的lammps模拟中&#xff0c;可能会对某一个特定晶粒的变形情况进行分析&#xff0c;此时&#xff0c;需要找到这个晶粒&#xff0c;并进行单独分析。 ovito有专用的晶粒识别命令&#xff0c;…...

DevEco Studio 3.1 Beta1版本发布——新增六大关键特性,开发更高效

智能代码编辑、端云一体化开发、低代码开发个性化…… 六大新增关键特性&#xff0c;开发更高效&#xff0c;体验更觉妙&#xff01; 立即点击链接下载&#xff0c;做DevEco Studio 3.1 Beta1版本尝鲜者&#xff01; 下载链接&#xff1a;HUAWEI DevEco Studio和SDK下载和升级 …...

【蓝桥杯每日一题】二分算法

&#x1f34e; 博客主页&#xff1a;&#x1f319;披星戴月的贾维斯 &#x1f34e; 欢迎关注&#xff1a;&#x1f44d;点赞&#x1f343;收藏&#x1f525;留言 &#x1f347;系列专栏&#xff1a;&#x1f319; 蓝桥杯 &#x1f319;我与杀戮之中绽放&#xff0c;亦如黎明的花…...

Spring Batch 高级篇-并行步骤

目录 引言 概念 案例 转视频版 引言 接着上篇&#xff1a;Spring Batch 高级篇-多线程步骤&#xff0c;了解Spring Batch多线程步骤后&#xff0c;接下来一起学习一下Spring Batch 高级功能-并行步骤 概念 并行步骤&#xff0c;指的是某2个或者多个步骤同时执行。比如下…...

对spring的@Cacheable缓存理解

1 什么是缓存第一个问题&#xff0c;首先要搞明白什么是缓存&#xff0c;缓存的意义是什么。对于普通业务&#xff0c;如果要查询一个数据&#xff0c;一般直接select数据库进行查找。但是在高流量的情况下&#xff0c;直接查找数据库就会成为性能的瓶颈。因为数据库查找的流程…...

力扣-市场分析

大家好&#xff0c;我是空空star&#xff0c;本篇带大家了解一道简单的力扣sql练习题。 文章目录前言一、题目&#xff1a;1158. 市场分析二、解题1.错误示范①提交SQL运行结果2.正确示范①提交SQL运行结果3.错误示范②提交SQL运行结果4.正确示范②提交SQL运行结果5.其他总结前…...

【2357. 使数组中所有元素都等于零】

来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 描述&#xff1a; 给你一个非负整数数组 nums 。在一步操作中&#xff0c;你必须&#xff1a; 选出一个正整数 x &#xff0c;x 需要小于或等于 nums 中 最小 的 非零 元素。nums 中的每个正整数都减去 x。 返回使 n…...

什么品牌的游戏蓝牙耳机比较好?玩游戏延迟低的蓝牙耳机推荐

游戏耳机的出现其实最主要的作用就是让玩家能够更专注的沉浸在游戏世界内&#xff0c;在声音层面去享受游戏的沉浸感&#xff0c;游戏最重要的就是操作灵敏&#xff0c;需要快速通过声音来判断敌人走向&#xff0c;所以小编特意整理了一期玩游戏延迟低的蓝牙耳机。 一、南卡小…...

day 33 状态压缩dp

二维状态压缩dp对于解决哈密顿回路问题的状态压缩dp只能计算固定起点到其他点的总方案数或最小路径等回路计数小蓝现在在第一栋教学楼&#xff0c;他想要访问每栋教学楼正好一次&#xff0c;最终回到第一栋教学楼&#xff08;即走一条哈密尔顿回路&#xff09;可看做&#xff1…...